The goal is to create an HTTP API for uploading, optimizing, and serving images.
- The API should expose an endpoint for uploading an image. After uploading, the image should be sent for optimization via a queue (e.g. RabbitMQ) to prevent excessive system load in case of many parallel image uploads and increase the system durability.
- Uploaded images should be taken from the queue one by one and optimized using the
github.com/h2non/bimg
go package (orgithub.com/nfnt/resize
package). For each original image, three smaller-size image variants should be generated and saved, with 75%, 50%, and 25% quality. - The API should expose an endpoint for downloading an image by ID. The endpoint should allow specifying the image optimization level using query parameters (e.g.
?quality=100/75/50/25
).
- Functionality. The developed solution should function as described in the "Summary" section. However, if you think that you can create a solution better than described in the "Summary" section, you are welcome to do so.
- Code simplicity. The architecture should be simple and easy to understand, the code should be well-formatted and consistent. Usage of code formatters (like gofmt) and linters (like golangci-lint) is encouraged.
Workflow:
App structure:
/internal
/app // Application that has all required components
/delivery // Package that include infrastructure for communicating by HTTP protocols
/http // http
/handler // user defined handler
/rest // REST API methods for handler
/server // http server
/rabbitmq // amqp
/client // RabbitMQ client
/domain // Domain business logic
/dto // Data Transfer Object
/repositories // Interfaces for services (use-cases)
/infrastructure // Actual implementation of components
/file // Local file storage (using standard pkg os / filepath / io/ioutil)
/worker // Background job / service that proceed the image from MessageBroker
/compressor // as a part of background job
/services // Services that App uses
/image // Image service
/publisher // Part of MessageBroker (only Send to...) for HTTP API
make rabbit // Up the RabbitMQ instance from docker
make rabbit-stop // Stop and delete the container
go run cmd/main.go
You can see the examples.log file for actual output of the application
Pay attention that in the log I've put the ID, URL and etc. data that are not for production!
It's a test task, so I just skip this part of hiding the USER information (but what someone can steal? ImageID? XD)
REMOVE the ID, URL and other information or CROP it
- WRITE UNIT/INTEGRATION/E2E TESTS (but it takes TOO MUCH time)
- Add database for register the errors in background job, if user wants to know which problems his/her image has
- Redirect errors from the Goroutines
- Remove HARDCODE and add ENV setup or flags of app
- USE the specific exchanger/channel/queue of RabbitMQ
- Use relative path
- Use caching (if need)
- . . .
I don't use the github.com/h2non/bimg
package due to its dependency for Linux.
I've written this application on Windows platform. I could use the WSL2 for example, but it to long for open it.
Hope it won't be a problem :)
I've using the Thunder client (rangav.vscode-thunder-client
) extension from VS Code to proceed HTTP requests
The HTTP methods collections that have been used placed there: collection.json
PAY attention: in POST method you need replace the path to the image file in the body
"body": {
"type": "formdata",
"raw": "",
"form": [],
"files": [
{
"name": "image",
"value": "<PATH TO IMAGE>"
}
]
},
For GET HTTP method:
"url": "localhost:8080/img/<UUID OF IMAGE>"
Use the json file for import:
- Name variables in camelCase.
- Name local variables in lower case, in order not to violate the principle of encapsulation (when it is small, it is not possible to import).
- Do not write excessive comments that only add noise to the code. TIP: Tips for writing comments from Clean Code.
- Go prefers large files.
http/rest/api/get.go
andhttp/rest/api/post.go
can be combined into one file. If other endpoints were to be added in the future, it would not be convenient to keep all GETs in one file and all POSTs in another. - Use a single style for error checking: First, Second.
domain/repositories // Interfaces for services (use-cases)
. Confusing, because there is pattern repository which is responsible for accessing the data.- If it were necessary to use a queue in another project with an implementation of RabbitMq, how would it happen?
- By design, the
infrastructure
level s responsible for implementations, but there is aCompressor
interface. It is better to create a separatecompressor
package with theresize
implementation as well as the queue. - The
infrastructure/worker/utils.go
file namedutils
is considered bad practice because it collects different functions from different places that could actually be put into separate packages without creating chaos in that file. It could be exported, for example, topkg/image
. - It is not obvious that in
Worker.Start()
the logic of working with the image is prescribed. It would be better if the worker was responsible only for receiving messages and transferring them to another handler, which he does not know about at all. That is, a callback would be passed toStart
, for example, which would call a service for image processing. - What are your business rules in the application? Where would you put these rules?
- Image quality validation occurs at the http level. If
ReadImageFromStorage
is not called via http, the validation will no longer work.
Queue refactor (refers to item 7):
- Create a separate `pkg/queue' package.
- Move the
MessageBroker
interface to thepkg/queue/queue.go
package (now it is indomain/repositories/queue/interface.go
). MessageDTO
is a generic message that should be sent by the queue and it is correct, but it should be moved closer to where it is used inpkg/queue/queue.go
(it is currently indomain/dto/dto.go
) .- Move the rabbitmq implementation to
pkg/queue/rabbitmq.go
. Now the queue can be easily reused.
- No errors from golangci-lint.
- There is a cool README, with a diagram and suggestions for improvement.
- There is a docker.
- The test is not abandoned by one commit. Although the name of the commits could be made more meaningful by using Conventional Commits.
- There are interfaces with further implementation.
- There is logging at all levels.
domain/repositories/queue
. The queue is transport. And this is correct, because data is not stored in the queue, data is simply passed through it in the same way as rest. Input-Output.