This sample project was created as a collection of the various things I've learned about best practices building microservices using Go. I structured the project using a hexagonal style abstracting away business logic from dependencies like the RESTful API, the Postgres database, and RabbitMQ message queue.
The Go community generally likes application directory structures to be as simple as possible which is totally admirable and applicable for a small simple microservice. I could probably have kept everything for this project in a single directory and focused on making sure it met twelve factors. But I'm a big fan of Domain Driven Design, and how it gels so nicely with Hexagonal Architecture and I wanted to see how a Go microservice might look structured using them.
The starting point of the application is under the cmd directory. The "domain" core of the application where all business logic should reside is under the core directory. The other directories listed there are each of the external dependencies for the project.
This project requires that you have Docker, Go and Make installed locally. If you do, you can start the application first by starting the docker-compose file, then start the application using the supplied Makefile.
docker-compose -f ./scripts/docker-compose.yml up -d
make run
If you want to create a deployable executable and run it:
make build
./bin/inventory
docker-compose up
This application uses the wonderful go-chi for routing beautifuly documentation served as the main inspiration for how to structure the API. Seriously, I was so impressed.
In Java I like to generate the controller layer using Open API so that the contract and implementation always match exactly. I couldn't quite find an equivalent solution I liked.
Truth be told, if I were doing inter-microservice communication I would strongly consider using gRPC rather than a RESTful API.
Many of the endpoints in this project are protected by using a simple authentication middleware. If you're interested in hitting them you can use basic auth admin:admin. Users are stored in the database along with their hashed password. Users are locally cached using golang-lru. In a production setting if I actually wanted caching I'd either use a remote cache like Redis, or a distributed local cache like groupcache to prevent stale or out of sync data.
This application outputs prometheus metrics using middleware I plugged into the go-chi router. If you're running locally check them out at http://localhost:8080/metrics. Every URL automatically gets a hit count and a latency metric added. You can find the configurations here.
I ended up going with zerolog for logging in this project. I really like its API and their benchmarks look really great too! You can get structured logging or nice human-readable logging by changing some configs
This project uses viper for handling externalized configurations. At the moment it only reads from the local config.yml but the plan is to make it compatible with Spring Cloud Config, and etcd.
I chose not to go with any of the test frameworks when putting this project together. I felt like using interfaces and injecting dependencies would be enough to allow me to mock what I need to. There's a fair bit of boilerplate code required to mock, say, the inventory repository but not having to pull in and learn yet another dependency for testing seemed like a fair tradeoff.
The testing in this project is pretty bare-bones and mostly just proof-of-concept. If you want to see some tests, though, they're in api. I personally prefer more integration tests that test an application front-to-back for features rather than tons and tons of tightly-coupled unit tests.
I'm using the migrate project to manage database migrations.
migrate create -ext sql -dir db/migrations -seq create_products_table
migrate -database postgres://postgres:postgres@localhost:5432/smfg-db?sslmode=disable -path db/migrations up
migrate -source file://db/migrations -database postgres://localhost:5432/database down
One of the goals of this service was to ensure all 12 principals of a 12-factor app are adhered to. This was a nice way to make sure the app I built offered most of what you need out of a Spring Boot application.
The application is stored in my git repository.
Go handles this for us through its dependency management system (yay!)
See the configuration section section above.
The application connects to all external dependencies (in this case, RabbitMQ, and Postgres) via URLs which it gets from remote configuration.
The application can easily be plugged into any CI/CD pipeline. This is mostly thanks to Go making this easy through great command line tools.
This app is not strictly stateless. There is a cache in the user repository. This was a design choice I made in the interest of seeing what setting up a local cache in go might look like. In a more real-world application you would probably want an external cache (like Redis), or a distributed cache (like Group Cache - which is really cool!)
This app is otherwise stateless and threadsafe.
The application binds to a supplied port on startup.
Other than maintaining an instance-based cache (see Process above), the application will scale horizontally without issue. The database dependency would need to scale vertically unless you started using sharding, or a distributed data store like Cosmos DB.
One of the wonderful things about Go is how fast it starts up. This application can start up and shut down in a fraction of the time that similar Spring Boot microservices. In addition, they use a much smaller footprint. This is perfect for services that need to be highly elastic on demand.
Docker makes standing up a prod-like environment on your local environment a breeze. This application has a docker-compose file that starts up a local instance of rabbit and postgres. This obviously doesn't account for ensuring your dev and stage environments are up to snuff but at least that's a good start for local development.
Logs in the application are written to the stdout allowing for logscrapers like logstash to consume and parse the logs. Through configuration the logs can output as plain text for ease of reading during local development and then switched after deployment into json structured logs for automatic parsing.
Database migration is automated in the project using migrate.
- Recreate architecture diagram
- Add godoc
- Return 204 no content if data already exists
- Cleanup TODOs