Some Thoughts About Software Architecture

Object-Oriented Semantics

Object Classes

Message Classes

Value Classes

Record Classes

Ports

Axiomatic Ports

Derived Ports

Adapters

Axiomatic Adapters

Derived Adapters

Comments

The terminology here is based on the terminology from Hexagonal Architecture (also called Ports and Adapters), but "derived" and "axiomatic" are being used instead of "primary" and "secondary" to emphasize the nature of the distinction.

The properties of a record should be orthogonal to each other, meaning that you can't infer anything about the value of one property based on the value of another. Otherwise there's a risk of inconsistency, and you need to go out of your way to validate the relevant invariants.

Adapters exist mostly to define input/output methods, and input/output methods should generally be asynchronous. This might be implemented with async/await in high-level languages. But in low-level languages, adapters are implemented with explicit state machines and message loops.

Aside from the current state of the state machine, and perhaps an immutable configuration, it's reasonable for adapters to be stateless. If there is a configuration, then it influences the set of interfaces that the adapter can reasonably implement. A configuration that exposes more properties, or more constrained values, can be used to implement more interfaces.

Adapters often expose methods that can change the state of a database. As a rule, those methods should be (a) idempotent and (b) transactional.

A distinction can be made between adapters that interact directly with the external world and those that don't. Adapters that interact directly with the external world must be written in a different language, because the runtime for a language cannot be implemented using the language itself.

Different adapters can exist at different layers of abstraction. Higher layers of abstraction should completely hide lower layers of abstraction, and they should reduce the exposed surface area in the process. This makes it easier to replace a given layer.

Messages are self-contained blobs of immutable state, and adapters are machines that change the world by pushing messages through pipes. A program that can't be understood in those terms, e.g. because messages sometimes push adapters through pipes, is very hard to reason about.

This isn't meant to be an exhaustive taxonomy of things-that-have-reasonable-semantics, but it is meant to be a step in that general direction.

Project Organization

Public Specification

Internal Specification

Internal Implementation

Public Implementation

Semantic Versioning