Font Awesome Free 5.13.0 by @fontawesome - https://fontawesome.com License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License)
Skip to main content

Boundaries and the database

How architectural boundaries apply to the use of databases

Boundaries between the domain and the database

Typically makes sense to draw a boundary between the actual domain logic and the database (unless your application is a thin layer around the database that doesn’t really have any domain logic)

One widespread convention: Repository pattern:

  • All interaction with the database is encapsulated inside Repository classes
  • The domain logic interacts with these classes, without having to know anything database-specific
interface UserRepository {
getUsers(): Promise<User[]>;
getUser(id: string): Promise<User>;
saveUser(user: User): Promise<void>;
deleteUser(id: string): Promise<void>;
}

class SqlServerUserRepository implements UserRepository {
// implement UserRepository methods by talking to SQL Server
}

class UserService {
constructor(private repository: UserRepository) { }

async updateName(id: string, newName: string) {
const user = await this.repository.getUser(id);
user.setName(newName);
await this.repository.saveUser(user);
}
}

If the domain logic is using the repository interface, then it also becomes easy to swap out the SqlServerUserRepository for a different implementation, for example an in-memory repository for testing purposes.

class InMemoryUserRepository implements UserRepository {
// implement UserRepository methods using in-memory storage
}

The Repository pattern also makes it easy to implement caching. Ideally, the repository takes care of caching values and invalidating the cache as needed, without other code even being aware that there is any caching at all.

Separation at the database level

For larger systems, it can make sense to separate different parts of the application down to the database level

  • Each part uses different tables or a different database
  • No direct links (like foreign keys) between data belonging to different parts
  • Considered good practice when setting up a microservices architecture
    • Sharing of database tables between services introduces tight coupling, potential data corruption in case of conflicting code between the services, ...
  • Can also do this in monolithic applications, potentially as a stepping stone towards a future microservices architecture
  • Easier to reason about separate parts of the application without having to think about other parts
  • More flexibility to change the schema or database technology for a certain part of the system

When drawing boundaries down to the database level, some data that is relevant to two parts of the system might exist on both sides of the boundary between them

See also Microservices, and specifically Microservices - Data duplication and bounded contexts

Resources

  • Clean Architecture (book by Robert C. Martin)
  • Building Evolutionary Architectures (book by Neal Ford, Rebecca Parsons and Patrick Kua) (summary slides )