/go-project-ddd-template

Template for GoLang services using concepts from domain driven design while trying to follow GoLang best practices

Primary LanguageGoMIT LicenseMIT

Go DDD Template

Domain Driven Design (DDD) Architecture template for Golang services

Go Report Card License Release codecov

Overview

The purpose of the template is to show:

  • how to organize a project and prevent it from turning into spaghetti code
  • where to store business logic so that it remains independent, clean, and extensible
  • how not to lose control when a microservice grows

Using the principles of the DDD community highlighting contributes from Eric Evans and Vaughn Vernon. Their work has had a significant impact on the way that many software developers approach building complex systems using DDD.

Go-ddd-template is created & supported by Sebastian Sigl.

Content

Quick start

Local development:

# Postgres, RabbitMQ
$ make compose-up
# Run app with migrations
$ make run

Integration tests (can be run in CI):

# DB, app + migrations, integration tests
$ make compose-up-integration-test

Libraries

This template comes with a set of libraries to quickly get up to speed.

GoLang httptest

Httptest is a package within the Go programming language that enables developers to test HTTP clients and servers more effectively. It provides utilities to create mock HTTP servers and clients, simulating real-world scenarios for thorough testing.

The httptest package is leveraged to execute system-tests efficiently, which are close to the main entry point, which is app_test.go.

Testcontainers

Testcontainers is a library for Golang that provides a friendly API to run Docker containers for integration testing (system-tests). It allows developers to spin up required services like databases, web servers, or any other application that can run in a Docker container as part of their test setup. This provides a consistent, reproducible, and isolated environment for tests, making it a powerful tool for modern test-driven development in Golang.

The system tests are located in internal/app_test.go and provide a high level coverage starting RabbitMQ and Postgres to run main use cases. Hence, makefile contains 2 tasks:

  • test, running all tests
  • test-fast, running all tests excluding slow running tests like system-tests

Test-Driven-Development and testcontainers

Testcontainers work seamlessly when they boot up swiftly. For containers that require more time, like the RabbitMQ testcontainer in this instance, you can utilize the reuse feature already activated in this template.

  1. Initiate a test that employs the necessary container, set a breakpoint, and maintain its operation as long as required.
  2. Begin code testing and implementation. Since containers are already operational, you'll use the running containers, accelerating the testing process.

To enhance convenience, a GitHub issue has been raised: testcontainers/testcontainers-go#1191

Cleanenv

Cleanenv is a utility library for reading and parsing configuration structures from files and environment variables. By importing the library and calling appropriate methods, you can easily manipulate and track your project's configuration. Additionally, Cleanenv can overwrite your configuration based on environment variables, providing flexibility for different execution contexts. The library also supports outputting a detailed list of configuration variables for easier debugging and tracking. Find the configuration definition and cleanenv integration in config/config.go.

Golang-migration

Golang-migrate is a powerful tool for managing database migrations, designed with the flexibility to function as either a CLI or library. With its focus on maintaining lightweight database drivers, it seamlessly integrates migrations from various sources and applies them in the correct sequence. It's designed with a fail-fast philosophy, meaning it will not make assumptions or attempt to correct user input, ensuring robust and reliable database migrations.

Use the makefile task migrate-up to run migrations.

For system tests, migrations are executed programmatically.

Avoiding application startup migrations:

Performing database migrations as part of the application startup can lead to problems, especially in production environments:

  1. Risk of Database Corruption and Downtime: Running migrations during startup can cause issues such as database corruption and downtime, particularly as the application scales.
  2. Parallel Migrations Issue: In production, multiple servers can attempt identical migrations simultaneously, potentially breaking the database.
  3. Downtime Due to Mental Coupling: If schema upgrades always occur during startup, you might assume that new code always runs with the latest schema. This could hinder rollbacks and cause downtime if a new code version is faulty.

Decoupling schema migrations from application startup allows for safer single migrations and reduces downtime risks. It also provides room for better testing of new code and supports zero-downtime migrations for high uptime requirements. It's important to adjust migration strategies based on whether you're working in a development or production environment.

Wire: Dependancy Injection Made Simple

Wire, created by Google, is a powerful and efficient compile-time dependency injection tool for Go. It's designed to simplify the task of initializing complex structures by automating the process. Wire works by generating a Go code file that resolves all of the dependencies in your project at compile time, ensuring type-safety, reducing runtime overhead, and enhancing code maintainability. It's an excellent choice for those striving for clean, robust, and efficient software design in the Go programming landscape.

Furthermore, it's important to maintain an emphasis on encapsulation and boundary enforcement. In some cases, this might mean not exposing a factory method. After all, while dependency injection is a powerful tool, it's not a mandate. It should be used judiciously, and not everywhere and always. Utilize dependency injection where it provides clear value, while maintaining the integrity of your software design principles.

Why Use Dependency Injection in Go?

Dependency Injection (DI) may initially seem at odds with Go's simplicity and "just enough" philosophy. However, DI can bring substantial advantages to your Go applications, especially when dealing with complex or large codebases. Here's why:

1. Enhances Code Testability:

Dependency Injection allows you to inject mock dependencies into your software during testing. This way, you can simulate various scenarios and focus on testing individual pieces of your system independently, promoting effective unit testing and enhancing overall code quality.

2. Improves Code Maintainability:

Dependency Injection decouples the relationship between dependent objects, making your code more modular. This decoupling allows you to change or replace dependencies without affecting other parts of your system. This makes it easier to update, maintain, and understand your code.

3. Facilitates Scalable Codebase:

As your Go application grows, manually managing dependencies can become daunting. DI automates the initialization and provision of dependencies, making your codebase more scalable and manageable.

4. Enables Effective Concurrency Handling:

In concurrent applications, managing object lifecycles can be challenging. Dependency Injection can manage object lifecycles, ensuring that each goroutine gets an appropriate instance (either a new one or a shared one based on the scope) of the required object.

5. Promotes Better Software Design:

Dependency Injection encourages you to think in terms of interfaces rather than concrete types. This leads to a design that is more flexible, extensible, and adheres to the SOLID principles, enhancing the overall software design quality.

With an API-First approach, it may not always fit to set up dependency injection for every aspect of your application. Yet, Wire provides us the flexibility to selectively manage and instantiate dependencies as needed. In this example generated code is partially disabled, because the required factory methods and generic monitoring endpoints are located in routers.go Most certainly therer is a smart solution for it, that we didnt figure out yet.

Project structure

cmd

The cmd directory is used to structure the application's executable binaries and their associated main functions. In the context of Domain-Driven Design (DDD), this directory aids in maintaining a clear separation of concerns, ensuring that the business logic, domain models, and infrastructure components remain decoupled from the application's entry points.

config

The config directory stores environment-specific settings and non-sensitive data. Avoid storing sensitive information like API keys or passwords within the source code or version control systems. The config directory facilitates a clean architecture by decoupling configuration details from domain logic. This approach allows developers to focus on the core business requirements.

For convenience, there is a .env.example file that can be used, which is set with default values for local usage.

The config structure is in the config.go. An env-required: true tag can be used to specify a value (either in yaml, or in environment variables).

Reading the config from .env contradicts the ideology of 12 factors, but in practice, it is more convenient than reading the entire config from ENV. It is assumed that default values are in .env, and security-sensitive variables are defined in ENV.

docs

The docs folder serves as a dedicated storage location for Swagger documentation generated by swag. It contains auto-generated API documentation files created by the Swag library, streamlining the development process and eliminating the need for manual updates.

integration-test

The integration-test folder is dedicated to housing integration tests, which are executed in an isolated container alongside the application container. This setup facilitates thorough testing of REST APIs through the use of the go-hit library, a powerful and user-friendly testing tool.

internal/app

There is always one Run function in the app.go file, which "continues" the main function.

This is where all the main objects are created. Dependency injection occurs through the "New ..." constructors (see Dependency Injection). This technique allows us to layer the application using the Dependency Injection principle. This makes the business logic independent from other layers.

Next, we start the server and wait for signals in select for graceful completion. If app.go starts to grow, you can split it into multiple files.

For a large number of injections, wire can be used.

internal/interfaces

Server handler layer (MVC controllers). The template shows 2 servers:

  • RPC (RabbitMQ as transport, located in internal/interfaces/rpc)
  • REST http (Gin framework, located in internal/interfaces/rest)

Server routers are written in the same style:

  • Handlers are grouped by area of application (by a common basis)
  • For each group, its own router structure is created, the methods of which process paths
  • The structure of the business logic is injected into the router structure, which will be called by the handlers

Instead of Gin, you can use any other http framework or even the standard net/http library.

In router.go and its handler methods, there are comments for generating swagger documentation using swag.

internal

The internal folder houses non-sharable code, safeguarding critical components of an application. It encompasses essential DDD (Domain-Driven Design) folders such as application, domain, and interfaces, tailored to each specific application. This structure ensures both optimal organization and robust security within high-scale software projects.

internal/domain

The internal/domain folder houses the crucial core domain code, serving as the foundation of our application's logic. This isolated directory ensures minimal external dependencies, promoting code integrity and maintainability. It is vital to the application, as it encompasses the essential logic that drives core functionalities.

internal/interfaces

The internal/interfaces folder houses crucial code for processing various input sources, such as CLI, REST APIs, and message-based systems. It standardizes data handling and streamlines integration across different communication channels. In Hexagonal Architecture terms, these files serve as incoming adapters, bridging external systems with the core application.

internal/application

The internal/application folder serves as a crucial bridge between external interfaces and internal domain and infrastructure components. It houses glue code that seamlessly integrates disparate elements to streamline application functionality. This organization allows for efficient grouping of related use cases within a single application service, enhancing maintainability and scalability

migrations

The migrations folder in Golang projects houses essential database migration files, facilitating schema updates and version control. These files contain SQL statements for creating, altering, or dropping tables and columns, allowing developers to synchronize database structure across environments. The migration process ensures consistent application behavior while reducing the risk of data corruption and enhancing collaboration among team members.

pkg

The pkg folder in Golang projects is a common convention for organizing shared code, often implementing to interfaces from the core domain. This structure promotes clean separation of concerns and facilitates code reusability across multiple services or applications.

Dependency Injection

In order to remove the dependence of business logic on external packages, dependency injection is used.

For example, through the New constructor, we inject the dependency into the structure of the business logic. This makes the business logic independent (and portable). We can override the implementation of the interface without making changes to the usecase package.

package usecase

import (
// Nothing!
)

type Repository interface {
  Get()
}

type UseCase struct {
  repo Repository
}

func New(r Repository) *UseCase {
  return &UseCase{
    repo: r,
  }
}

func (uc *UseCase) Do() {
  uc.repo.Get()
}

It will also allow us to do auto-generation of mocks (for example with mockery) and easily write unit tests.

We are not tied to specific implementations in order to always be able to change one component to another. If the new component implements the interface, nothing needs to be changed in the business logic.

To manage the effort and complexity to instantiate and inject dependencies, wire is used to generate factories at build-time.

DDD Architecture

Key idea

Programmers realize the optimal architecture for an application after most of the code has been written.

A good architecture allows decisions to be delayed to as late as possible.

The main principle

Dependency Inversion (the same one from SOLID) is the principle of dependency inversion. The direction of dependencies goes from the outer non-domain layers to the inner domain layer. Due to this, business logic and entities remain independent from other parts of the system.

You might know this from clean architecture or onion-architecture already.

Onion Architecture

The foundation of onion architecture and ddd is similar: both manage dependencies to put the domain logic in the center. Domain-driven-design contains much more guidance but on a package level, it's only about 4 layers:

DDD Architecture

We put business logic into the Domain layer, and wrap it with an Application layer, each with distinct functions:

  1. Domain Layer: Executes the fundamental, use-case agnostic business logic within the domain/system.
  2. Application Layer: Carries out application-specific use cases and contains IO heavy operations like fetching and storing data. It should contain as less logic as possible and acts as the glue between incoming access and the actual domain code to avoid domain leakage to outer layers.
  3. Interfaces Layer: Comprises UI components, REST-Controller, message-receivers and others incoming sources for the application.
  4. Infrastructure Layer: Bolsters other layers by implementing abstractions and integrating third-party libraries and systems.

Access flow starts from interfaces (e.g. REST), which access application services that are a facade for a specific group of use cases. The application service is a great place to handle transactions, logging, and other pure technical requirements. To keep it as free from domain logic as possible, a typical method fetches an entity, calls an operation on it, or via another service, and stores it.

The infrastructure layer does its work behind the scenes. If there is a repository that gets the data, the interface for the repository is located in the domain package because the domain requires it to operate. The implementation is located in infrastructure because from a domain point of view, it's not important how the data is fetched or stored. It's a detail that is abstracted away, which comes with a lot of benefits like testability and portability.

It's not always beneficial to create an interface in the domain and implement it in infrastructure. Sometimes, you just don't want to pollute the domain with something specific like monitoring. Hence, you may want to put the interface in the application layer.

For example, let's go through an example for access to the database from a REST controller.

The REST-Controller is part of the outer incoming interfaces layer. The database is part of infrastructure, which means they know nothing about each other.

The communication between them is carried out through use-case specific methods of the application service:

    REST > ApplicationService
           ApplicationService > repository (Postgres)
           ApplicationService < repository (Postgres)
    REST < ApplicationService

The symbols > and < show the intersection of layer boundaries through Interfaces.

Or more complex business logic:

    REST > ApplicationService
           ApplicationService > repository
           ApplicationService < repository
           ApplicationService > webapi
           ApplicationService < webapi
           ApplicationService > RPC
           ApplicationService < RPC
           ApplicationService > repository
           ApplicationService < repository
    REST < ApplicationService

Layers

DDD Architecture Terminology

  • Domain: A sphere of knowledge or activity that represents the business or system being modeled. It includes core concepts, business rules, and policies. Domain services and entities are located in the internal/domain folder.

  • Bounded Context: A well-defined boundary within the domain, encapsulating a specific area of concern and its related elements, helping to avoid ambiguity and maintain consistency.

  • Ubiquitous Language: A common language used by both technical and non-technical stakeholders to ensure clear communication and shared understanding of the domain concepts and requirements.

  • Entity: A domain object with a unique identity that represents a business concept and persists over time, encapsulating both state and behavior.

  • Value Object: An immutable domain object that represents a specific value or attribute, rather than a unique entity, and is defined by its properties.

  • Aggregate: A cluster of domain objects (entities and value objects) that are treated as a single unit, ensuring consistency and enforcing business rules.

  • Aggregate Root: The main entity within an aggregate that acts as a gateway for all interactions with the aggregate, managing its consistency and state.

  • Repository: An abstraction that provides methods for retrieving, storing, and updating aggregates, decoupling the domain from the underlying data storage mechanism.

  • Domain Event: A message or event that represents a significant occurrence within the domain, allowing for decoupled and reactive system design.

  • Domain Service: A stateless service that encapsulates domain logic that doesn't naturally fit within an entity or value object, coordinating interactions between domain objects.

  • Application Service: A service that coordinates the use of domain objects, repositories, and domain services to implement business use cases, acting as a bridge between the domain and the external world. Application Services are located in internal/application.

  • Infrastructure: The technical components and services that support the domain, such as databases, messaging systems, and external API integrations. Implementations are located in internal/infrastructure.

Balancing Test Strategies: The Effective Use of Mocks, Fakes, and Real Implementations

System Tests for Full Application Validation

System tests deliver a holistic view of the application's performance by simulating the entire endpoint execution process. These tests employ a docker-based database, which launches quickly, and fake service endpoints to simulate actual operations. The primary aim of these tests is to detect high-level configuration or component integration issues that could impede the application's functionality in a production environment. These tests could also fail due to class-level issues, which, while acceptable, should ideally be caught by parallel running tests.

Narrowly Scoped Tests for Detailed Business Logic

The intricacies of business logic, from validation rules to exception cases, necessitate rigorous testing. The scope of these tests is limited to a few classes to ensure swift execution and precise identification of potential problems.

Integration Tests between Application and External Systems

Running integration tests with external systems, using either a docker-based database or a fake server, proves invaluable in understanding system interactions, particularly in the context of database interactions and repository writing tests. Though slower than system tests, they provide a more comprehensive view of system interoperability.

In conclusion, while additional tests may fill the spaces between these broad categories, these three levels of testing provide a robust starting point for achieving a balanced testing environment. By determining the right approach (mocking, fakes, or real environment) for the task at hand, you can ensure efficient and effective test coverage.

Interface-Type

Maintaining interface-types between the domain layer and infrastructure is crucial for inverting dependencies, which is especially beneficial in GoLang for preventing circular dependencies. Interface-types also enhance extensibility, adhering to the Open-Close principle, and improve testability by simplifying dependency stubbing. While interface-types between all layers add value, they also introduce additional code to maintain and increase complexity to comprehend.

Alternative approaches

In addition to DDD architecture, Onion architecture, Hexagonal (Ports and adapters) or Clean architecture are similar to it. All are based on the principle of Dependency Inversion and separating and avoid dependencies to the business logic.

Similar projects

Useful links