This repository contains a very simple example project to explore the application of a (monolithic) Ports and Adapter Pattern / Hexagonal Architecture. It uses golang (1.18 with generics) for it's implementation.
Tries to give an answer if in an example code one can find: "error handling omitted for simplicity".
The service represents a very simple URL shortener service. Offers basic CRD (create, read, delet/invalidate) operations via REST.
Based on the series from tensor-programming and recommendations from:
- How do you structure your Go apps? - Kat Zień
- Improving the code from the official Go RESTful API tutorial - Ben Hoyt
The implementation in this repository could be a little is over-engineered for such a simple project. It exists mostly for its educational purpose and as an experiment to break with traditional approaches (e.g. the active record pattern, ORM, storing JSON in redis, coupling of the domain and the entities).
This repository is intended for educational purposed. By advocating a certain architecture style (see below), it offers multiple implementations for specific aspects in the final binary. As a side effect it requires dependencies (e.g. sqlite with CGO) even if an option is not used via the configuration. The following aspects can be configured on start-up time:
- the router, takes no addition attributes and shall be one of the following:
- go: stdlib only router
- chi: chi router
- httprouter: httprouter
- gorilla: gorilla/mux
- gin: gin uses an alternative implementation for
http/rest
due to the different method signatures of the handlerFunc
- the repository, specifies the dsn (Data Source Name)
It can be configured either by a shortener.env
file or by setting the environment variables directly.
Memory backed (great for testing):
repository=memory
Plain sqlite repository:
repository=sqlite://file::memory:?cache=shared&_journal_mode=WAL&_foreign_keys=true
Gorm (ORM) + an sqlite repository:
repository=gormsqlite://file::memory:?cache=shared&_journal_mode=WAL&_foreign_keys=true
As of today (January, 2022), golang 1.18 with generics is not yet released, but betas are available. Please refer to: https://go.dev/blog/go1.18beta2
go install golang.org/dl/go1.18beta2@latest
go1.18beta2 download
-
Either in Codium / VS Studio Code: "Go: Install/Update Tools"
-
Or manually. Example for POSIX based systems:
mkdir /tmp/gopls && cd "$_"
gotip mod init gopls-unstable
gotip get golang.org/x/tools/gopls@master golang.org/x/tools@master
gotip install golang.org/x/tools/gopls
- View > Command Palette
- Go: choose Go environment
- select go1.18beta2
The project follows the recommendations from Kat Zień how to structure golang programs in the "domain" fashion. Please refer to her talks about the reasoning why separating services into own folders.
As of now, most software is (still) developed by humans (or synthetics like copilot) in the form of text files or meshed boxes (e.g. "no-code", node-red). The software is then somehow interpreted and executed by a computing instance (CPU, GPU). While the computing instance does not care about the structure of the software, the ones responsible for its development or maintenance (should) care. There is no clear distinction between the terms: software design and software architecture, but both describes how a software is structured in the small and in the large. The terms became even more funny if "common characteristics of good examples" (Alistair in the "Hexagone") became "design patterns" (thanks to Christopher Alexander). One can spent a whole career in this domain (e.g. https://www.martinfowler.com/architecture/).
There are several publications and technologies around architecture/design/patterns like: the POSA series, MVC, MVVC, about React, layered architecture, SOA, ESB, micro services, ...
A (not so) recent addition to the attempt to find the holy grail a way for: "faster delivery of new features" (fowler), is the ports and adapter architecture/pattern (https://web.archive.org/web/20060711221010/http://alistair.cockburn.us:80/index.php/Hexagonal_architecture) by Alistair Cockburn from 2005. It's better known by its working title "Hexagonal Architecture".
In the end it boils down to: "ports are interfaces" (Alistair in the "Hexagone") and the name was chosen "because it had to be a noun" and:
The hexagon is intended to visually highlight:
(a) the inside-outside asymmetry and the similar nature of ports, to get away from the one-dimensional layered picture and all that evokes, and
(b) the presence of a defined number of different ports - two, three, or four (four is most I have encountered to date).
The hexagon is not a hexagon because the number six is important, but rather to allow the people doing the drawing to have room to insert ports and adapters as they need, not being constrained by a one-dimensional layered drawing. The term hexagonal architecture comes from this visual effect.
The hexagon architecture follows the input -> compute -> output
pattern, except that it's called: primary actors -> application -> secondary actors
. The parts are accessible via ports/interfaces.
Another architectural style was defined by Jeffrey Palermo a few years later in 2008 called the Onion Architecture. It's not longer shaped as a hexagon, but as concentric circles and resembles the UNIX architecture (one of many diagrams).
"This architecture is unashamedly biased toward object-oriented programming, and it puts objects before all others." according to its creator. The basic idea that in its center lies the Domain Model (Domain-Driven Design, Eric Evans, 2004) that represents the state and behavior of the application. Each additional layer encloses the domain and adds more behavior. The first layer around the Domain Model provides interfaces for saving and retrieving data (Objects) via a repository. The interface decouples the Domain Model from the concrete database which resides outside of the Domain Model.
Hexagonal architecture and Onion Architecture share the following premise: Externalize infrastructure and write adapter code so that the infrastructure does not become tightly coupled.
- Steve Francia - Go: building on the shoulders of giants and stepping on a few toes
- Rob Pike - Simplicity is Complicated
- Dave Cheney - SOLID Go Design
- implement and test mongo backend
- implement the code generator that creates the conversion code that performs the conversion without runtime inspection (reflection)
- dockerize (also for macOS)
- docker-compose with different storage backends
- time to live (ttl)
- top10 (update on read)
- internal event sourcing to simulate Command and Query Responsibility Segregation (CQRS)?
If the docker image is build behind a corporate proxy, one can either use the global instance or a local proxy running on localhost (e.g. CNTLM). Either way, the proxy can be configured with the environment variable CONTAINER_HTTP_PROXY
and CONTAINER_HTTPS_PROXY
. The Task process task docker
will set the proxy variables based on this variables.
If the setup is by coincidence Windows 10 with WSL2 and docker installed inside a WSL distro (not the docker desktop client), placing the following code in the .bashrc
:
IP="$(ip -o -4 a | awk '$2 == "eth0" { gsub(/\/.*/, "", $4); print $4 }')"
export CONTAINER_HTTP_PROXY_SERVER="${IP}"
export CONTAINER_HTTP_PROXY_PORT="3128"
export CONTAINER_HTTP_PROXY="http://${IP}:${CONTAINER_HTTP_PROXY_PORT}"
export CONTAINER_HTTPS_PROXY="http://${IP}:${CONTAINER_HTTP_PROXY_PORT}"
This snippet provides the environment variables CONTAINER_HTTP_PROXY
and CONTAINER_HTTPS_PROXY
for the interactive shell with the current IP address of the WSL2 host. It seams like that the IP addresses are not fixed and change from startup to startup. This makes calling task docker
transparent if a proxy is used or not.
If the corporate proxy is an intercepting man-in-the-middle (MITM) proxy, the secure connection can't be verified during downloading the dependencies. Instead of ignoring security completely, the certificate of the MITM-proxy must be added to the docker container. Conveniently placing the *.crt
certificate files in the certificates
folder will automatically used during build. These certificates will not be available in the final docker image.
docker run --rm -it -p 8000:8000 --env "BIND=:8000" crra/hex-microservice
NOTE: The --env "BIND=:8000"
is required due to the default bind value which binds to localhost only.