Go-kit Foundation
This is an example base implementation of a go-kit project.
Pre-requisites
Docker Dev (preferred)
For Docker, we have chosen to employ a two-step build and run container procedure
for our Go projects.
To use the gokit-base docker container you'll first need to build a container
using the scripts supplied in the docker/
folder in this repository. On your first
run you'll need to execute ./docker/build.sh -i -v latest
.
Finally, docker-compose up
Local Dev
To get the gokit-base project up and running you'll need to have a few things installed beforehand:
- Install Go
- Install Consul
Before building the sample application you will need to bootstrap Consul with
environment configuration and export necessary variables to initialize the app.
The environment configuration example can be found in the docker/gokit-base/resources
folder. The application's KV path where you will need to PUT to can be found in
config/config.go
.
Finally, run go build
and ./gokit-base
to start the listening server!
Repository and Project Structure
app
│ .gitignore
│ docker-compose.yml
│ Jenkinsfile
│ main.go
│ README.md
│
└───config
│ │ config.go
└───docker
│ │ build.sh
│ └───app
│ │ │ Dockerfile
│ │ └───bin
│ │ │ │ .gitkeep
│ │ │ │ app_binary
│ │ │ │ ...
│ │ └───resources
│ │ │ │ env.default
│ │ │ │ init.sh
│ │ │ │ ...
│
└───domain_object
│ │ domain_object.go
│ │ endpoint.go
│ │ instrumenting.go
│ │ logging.go
│ │ service.go
│ │ transport.go
└───health
│ │ endpoint.go
│ │ instrumenting.go
│ │ transport.go
└───vendor
Enumeration of the Project Structure
- docker-compose.yml describes the composition of application container(s) and
dependencies as well as any networking considerations.
- Jenkinsfile contains Jenkins 2 pipeline configuration scripts written in Groovy for CI/CD.
- The
config/
folder contains your specific application configuration code. This may be something as simple
as an Environment struct read from Consul or MySQL connection details (also read from Consul). We are currently
using Viper to manage configuration and an example can be found in this project.
- The
docker/
folder contains your application container Dockerfile, container build and runtime configuration in build.sh
and
init.sh
, and an ignored binary folder that will be the target of the container build script.
- The
domain_object/
folder is a sample folder structure for an application business object. In the example app
this can be seen in the users/
folder. Further discussion on go-kit idioms such as endpoint.go
will follow.
- The
health/
folder includes an example basic HTTP health endpoint. This folder may grow to contain other application
health checks including canaries or integration health checks.
- The
vendor/
folder is not committed to source control, but shown here to demonstrate the location of installed
vendor libraries.
Go-kit Fundamentals
This section will be a brief, high level discussion of the basic go-kit idioms employed in this example repository.
A better (and more thorough) resource can be found at Peter Bourgon's website or the go-kit repo itself which includes
a thorough example application structured from the Domain Driven Design book "Shipping" topic:
Endpoints are "the building block of Go-kit components". They are "implemented by servers, and called by
clients". An endpoint ingests an application service and decodes and encodes transport requests/responses. Requests
and responses should have a defined struct
that will contain data parsed from the client request or from the application
service for the response. Endpoints are chainable and may be wrapped by Middleware. An example of this kind of
decoration may be observed in the logging.go
or instrumenting.go
files, although these wrap Service methods.
Transports are "bindings" to "concrete" transport methods. Simply put, this means that your transport.go
file should
contain any HTTP, gRPC, or socket transport logic. If you are coming from another micro framework, this may be a
unique practice. Typically we see business service logic (the stuff in service.go
) in a handler function. Go-kit
encourages us to extract that logic and place it in another module and keep our transport specific code pristine.
Loggers like the one you may find in logging/logger.go
implement Go-kit's application logging conventions. This package
"may be wrapped to encode conventions, enforce type-safety, provide leveled logging, and so on". In the example application
we wrap the Go-kit logger to enable Monolog-style formatted logs for our Heka decoders to consume. In this manner we are able
to define a single safe format for all of our applications' logs. Telemetry works in the same way.
Final Notes and Caveats
This example app is meant to be a template for hardened, production ready Go microservices. By no means
is this a "one size fits all" template. There may be very small services that don't require the structure defined
here BUT we have an answer for that too! All of these constructs are easily chainable and reasonable to place
in a simple main.go
file alongside some other helper modules and a Dockerfile.
Additionally, we don't claim to have all of the answers! If you have a better idea or practice please open an Issue
against this repository with your suggestion and we will do our best to facilitate a discussion about your concern or suggestion.
Thanks!