2020-06-07 14:26:06 +02:00
# Iris, MySQL, Groupcache & Docker Example
## 📘 Endpoints
| Method | Path | Description | URL Parameters | Body | Auth Required |
|--------|---------------------|------------------------|--------------- |----------------------------|---------------|
| ANY | /token | Prints a new JWT Token | - | - | - |
2020-11-25 03:47:36 +01:00
| GET | /category | Lists a set of Categories | offset, limit, order | - | Token |
2020-06-07 14:26:06 +02:00
| POST | /category | Creates a Category | - | JSON [Full Category ](migration/api_category/create_category.json ) | Token |
| PUT | /category | Fully-Updates a Category | - | JSON [Full Category ](migration/api_category/update_category.json ) | Token |
| PATCH | /category/{id} | Partially-Updates a Category | - | JSON [Partial Category ](migration/api_category/update_partial_category.json ) | Token |
2020-11-25 03:47:36 +01:00
| GET | /category/{id} | Prints a Category | - | - | Token |
2020-06-07 14:26:06 +02:00
| DELETE | /category/{id} | Deletes a Category | - | - | Token |
2020-11-25 03:47:36 +01:00
| GET | /category/{id}/products | Lists all Products from a Category | offset, limit, order | - | Token |
2020-06-07 14:26:06 +02:00
| POST | /category/{id}/products | (Batch) Assigns one or more Products to a Category | - | JSON [Products ](migration/api_category/insert_products_category.json ) | Token |
2020-11-25 03:47:36 +01:00
| GET | /product | Lists a set of Products (cache) | offset, limit, order | - | Token |
2020-06-07 14:26:06 +02:00
| POST | /product | Creates a Product | - | JSON [Full Product ](migration/api_product/create_product.json ) | Token |
| PUT | /product | Fully-Updates a Product | - | JSON [Full Product ](migration/api_product/update_product.json ) | Token |
| PATCH | /product/{id} | Partially-Updates a Product | - | JSON [Partial Product ](migration/api_product/update_partial_product.json ) | Token |
2020-11-25 03:47:36 +01:00
| GET | /product/{id} | Prints a Product (cache) | - | - | Token |
2020-06-07 14:26:06 +02:00
| DELETE | /product/{id} | Deletes a Product | - | - | Token |
## 📑 Responses
* **Content-Type** of `"application/json;charset=utf-8"` , snake_case naming (identical to the database columns)
* **Status Codes**
* 500 for server(db) errors,
* 422 for validation errors, e.g.
```json
{
"code": 422,
"message": "required fields are missing",
"timestamp": 1589306271
}
```
* 400 for malformed syntax, e.g.
```json
{
"code": 400,
"message": "json: cannot unmarshal number -2 into Go struct field Category.position of type uint64",
"timestamp": 1589306325
}
```
```json
{
"code": 400,
"message": "json: unknown field \"field_not_exists\"",
"timestamp": 1589306367
}
```
* 404 for entity not found, e.g.
```json
{
"code": 404,
"message": "entity does not exist",
"timestamp": 1589306199
}
```
* 304 for unaffected UPDATE or DELETE,
* 201 for CREATE with the last inserted ID,
* 200 for GET, UPDATE and DELETE
## ⚡ Get Started
Download the folder.
### Install (Docker)
Install [Docker ](https://www.docker.com/ ) and execute the command below
```sh
2020-11-25 03:47:36 +01:00
$ docker-compose up --build
2020-06-07 14:26:06 +02:00
```
### Install (Manually)
2021-02-19 07:49:15 +01:00
Run `go build -mod=mod` or `go run -mod=mod main.go` and read below.
2020-06-07 14:26:06 +02:00
#### MySQL
Environment variables:
```sh
MYSQL_USER=user_myapp
MYSQL_PASSWORD=dbpassword
MYSQL_HOST=localhost
MYSQL_DATABASE=myapp
```
2020-11-24 13:58:02 +01:00
Download the schema from [migration/db.sql ](migration/db.sql ) and execute it against your MySQL server instance.
2020-06-07 14:26:06 +02:00
```sql
CREATE DATABASE IF NOT EXISTS myapp DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
USE myapp;
SET NAMES utf8mb4;
SET FOREIGN_KEY_CHECKS = 0;
DROP TABLE IF EXISTS categories;
CREATE TABLE categories (
id int(11) NOT NULL AUTO_INCREMENT,
title varchar(255) NOT NULL,
position int(11) NOT NULL,
image_url varchar(255) NOT NULL,
created_at timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
updated_at timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
PRIMARY KEY (id)
);
DROP TABLE IF EXISTS products;
CREATE TABLE products (
id int(11) NOT NULL AUTO_INCREMENT,
category_id int,
title varchar(255) NOT NULL,
image_url varchar(255) NOT NULL,
price decimal(10,2) NOT NULL,
description text NOT NULL,
created_at timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
updated_at timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
PRIMARY KEY (id),
FOREIGN KEY (category_id) REFERENCES categories(id)
);
SET FOREIGN_KEY_CHECKS = 1;
```
### Requests
Some request bodies can be found at: [migration/api_category ](migration/api_category ) and [migration/api_product ](migration/api_product ). **However** I've provided a [postman.json ](migration/myapp_postman.json ) Collection that you can import to your [POSTMAN ](https://learning.postman.com/docs/postman/collections/importing-and-exporting-data/#collections ) and start playing with the API.
All write-access endpoints are "protected" via JWT, a client should "verify" itself. You'll need to manually take the **token** from the `http://localhost:8080/token` and put it on url parameter `?token=$token` or to the `Authentication: Bearer $token` request header.
### Unit or End-To-End Testing?
Testing is important. The code is written in a way that testing should be trivial (Pseudo/memory Database or SQLite local file could be integrated as well, for end-to-end tests a Docker image with MySQL and fire tests against that server). However, there is [nothing(?) ](service/category_service_test.go ) to see here.
## Packages
2020-11-24 13:58:02 +01:00
- https://github.com/kataras/jwt (JWT parsing)
2020-06-07 14:26:06 +02:00
- https://github.com/go-sql-driver/mysql (Go Driver for MySQL)
- https://github.com/DATA-DOG/go-sqlmock (Testing DB see [service/category_service_test.go ](service/category_service_test.go ))
- https://github.com/kataras/iris (HTTP)
- https://github.com/mailgun/groupcache (Caching)