Cloud native service to generate thumbnails from min.io.
First, a few packages are added and initialized. I use viper for configuration and zap logger for logging.
I added the docker setup for local development. The project will use the underlying libvips library. To make sure all developers get the same library and avoid "works on my machine", let´s develop inside the container. One catch tough: for unknown reasons the IDE GOLAND does not support Go runtimes inside Docker. There are only build targets with Docker supported. So make sure you check that the GoLand > Preferences > Go runtime is compatible with this project. Which is build on Go 1.16.
cd _INFRA/dev && docker-compose up -d
docker exec -it dev_thumbnail-generator_1 bash
go run .
What we built so far: branch
Next step: Add min.io. The easiest way to get started it checking the (GUI)[http://localhost:8080/login]. Create the bucket "app-images" here. Now at _INFRA/dev/minio-data a new directory should exist. Now get an image. Why not this one from (unsplash)[https://unsplash.com/photos/sAVFADKItCo] Either upload it via the min.io web interface or just drop it in the "minio-data/app-images" directory. Your bucket contains one image now. Now your config/local.yaml file should contain the absolute path to the local download directory. This is inside the container so the location is /app/downloaded-images.
Now we are finally ready for a test drive. Let´s download an image from min.io and store it locally. I added a
simple fetch operation to main.go.
Run go run .
What we built so far: branch
Now we need to talk a bit what is going to happen next. Let´s assume the application knows the name of the image already. These are the steps that need to be performed:
- Download image from min.io and save it locally to originals directory
- Generate image thumbnail and save it to thumbnails directory
- Upload thumbnail to "thumbnails" path in bucket in min.io
- Delete both local files (original / thumbnail)
Download and upload are happening on the same bucket hence there is no need for a separation of services. My approach would be: building a service for all min.io operations that are needed: Download, Upload, Delete local file and a few helper. Then use a facade as a client of that service. Now that we have everything together interfaces need to be defined, struct written and injected from main.go.
This took quite a while and quite a bit code changed. It makes sense to see if everything is working. Now the image should still be downloaded and end up in the download-images/originals directory.
What we built so far: branch
We´re getting closer. Next the wrapper for the thumbnail generator is going to get added. We chose
libvips. I won´t go into detail here but reason for
that was the great research and benchmarking done here speedtest-resize.
It is possible to crop files in natively in Go, or use a C library with CGO or execute a binary. The latter being the
most efficient option. One caveat here: when you want to keep the option open to use this as an AWS Lambda function
this might not be a good choice. In my own use case this was different: we control the underlying container image
hence there are no problems with external dependencies.
I added the next steps. Thumbnails get generated, uploaded to min.io and both local files deleted.
What we built so far: branch
The image processing workflow is now setup. We process just one file so far. On the next step we need a way the application receives the image name to be processed. Several options seem possible:
- Mini.io events and an event loop listener via sdk
- Mini.io webhook that sends data to a http endpoint
- Sending events from the backend application to a http endpoint
The first option with the mini.io events sounds striking first. There´s a big catch tough. You will lose the ability to have multiple replicas of the service. Unlike Consumers in a Consumer Group on an Apache Kafka topic every event listener consumes the same events. The service is supposed to live inside a Kubernetes cluster. When scaling up you end up with multiple replicas all producing the same images. This is not how scalability should work. So sticking to the http option is probably the best way. You can have multiple replicas of your pod and either your kubernetes service for the deployment or Ingress load balancer makes sure the workload is distributed evenly on all pods.
So a web server it is. We are going to build one. What we need is a HandlerFunc that holds all logic which is mainly:
parsing and validating the JSON request. Call process facade. Send back a response. There needs to be middleware as well
to make sure the method is POST and the content is JSON. All this is basically "how to build a web server from scratch".
If you have no clue what is going on here I recommend Alex Edward´s book "Let´s go". This is one of the best
introductions to web development with Go. The first step is the server in main.go. Here a default router
http.NewServeMux()
is set and routes are added. The application has one route and needs additional liveness and
readiness probes for kubernetes. The "create" route needs both middlewares we talked about. These are wrappers around
the HandlerFunc.
The process facade now is getting moved to the controller method. In this setup the request is parsed and the receiver immediately receives a response. The actual image processing is done in a Goroutine in the background.
The server runs inside the Docker container. As we opened port 3000 it should be accessible from your host machine. So it can be reached via Curl or Postman. I have one image in my min.io bucket. The app should produce a cropped version of that image and upload it to minio in the thumbnails folder.
curl --location --request POST 'localhost:3000/create' \
--header 'Content-Type: application/json' \
--data-raw '{
"fileName": "livia-sAVFADKItCo-unsplash.jpg"
}'
There is also validation added with unique rules. As not all files are image types that can be processed only names with certain image extensions are valid payloads for the endpoint. Don´t expect your .doc files to be processed.
What we built so far: branch
Added a Dockerfile for the final build. docker build -t go-thumbnails:latest .