1
0
mirror of https://github.com/raseels-repos/golang-saas-starter-kit.git synced 2025-06-06 23:46:29 +02:00

313 lines
10 KiB
Markdown
Raw Permalink Normal View History

# SaaS Web API
Copyright 2019, Geeks Accelerator
twins@geeksaccelerator.com
## Description
Web API is a client facing API. Standard response format is JSON. The example web-api service includes API documentation.
While the web app is meant for humans to experience and requires a friendly UI, the web API is meant for customers or
third-party partners of your SaaS to programmatically integrate. To help show the similarities and differences between
the pages in the web app and similar endpoints in the web API, we have created this diagram below. Since it is
very detailed, you can click on the image to see the larger version.
[![Diagram of pages in web app and endpoints in web API](resources/images/saas-starter-kit-pages-and-endpoints-800x600.png)](https://gitlab.com/geeks-accelerator/oss/saas-starter-kit/tree/master/resources/images/saas-starter-kit-pages-and-endpoints-800x600.png)
This web-api service is not directly used by the web-app service to prevent locking the functionally required for
internally development of the web-app service to the same functionality exposed to clients via this web-api service.
This separate web-api service can be exposed to clients and be maintained in a more rigid/structured process to manage
client expectations.
**Not all CRUD methods are exposed as endpoints.** Only endpoints that clients may need should be exposed. Internal
services should communicate directly with the business logic packages or a new API should be created to support it. This
separation should help decouple client integrations from internal application development.
## Making Requests to Web API
2019-08-16 15:50:07 -03:00
Once the web-api service is running it will be available on port 3001.
http://127.0.0.1:3001/
The easiest way to make requests to the web-api service is by using CURL via your CLI.
To make a request to the web-api service you must have an authenticated user. Users can be created with the API but an
initial admin user must first be created. The initial admin user can easily be created using the Swagger API documentation.
## API Documentation
Documentation for this API service is automatically generated using [swag](https://github.com/geeks-accelerator/swag).
The Swag Go project also provides a web UI to allow you and your customers of your SaaS to explore your API - its exposed
business logic - as well as easily try our that exposed functionality.
Once this web-api service is running, the Swagger API documentation for the service can be accessed at /docs:
http://127.0.0.1:3001/docs/
You can refer to the example of the API documentation that we have deployed on production for you here:
https://api.example.saasstartupkit.com/docs/
[![Example Golang web app deployed](https://dzuyel7n94hma.cloudfront.net/saasstartupkit/assets/images/responsive/img/saas-startup-example-golang-project-web-api-documentation-swagger-ui.png/9334f34bf028e0656f73aeb9d931e726/saas-startup-example-golang-project-web-api-documentation-swagger-ui-320w-480w-800w.png)](https://api.example.saasstartupkit.com/docs/)
## Local Installation
### Build
```bash
go build .
```
### Docker
2019-08-16 15:50:07 -03:00
To build using the docker file you need to be in the project root directory.
The `Dockerfile` assumes the working context is the root directory since it references Go Module
files to build the image.
```bash
docker build -f cmd/web-api/Dockerfile -t saas-web-api .
```
2019-06-24 22:41:21 -08:00
## Getting Started
2019-08-16 15:50:07 -03:00
### 1. Ensure dependant services are running.
Navigate to the project root where `docker-compose.yaml` exists. There is only
one `docker-compose.yaml` file that is shared between all services.
*Start Services.*
```bash
docker-compose up -d
```
2019-08-16 15:50:07 -03:00
#### 1.1. Set env variables.
2019-08-16 15:50:07 -03:00
Copy the sample file to make your own copy:
```bash
cp sample.env local.env
```
2019-08-16 15:50:07 -03:00
Make any changes to your copy of the file if necessary and then add them to your env:
```bash
source local.env
```
2019-08-16 15:50:07 -03:00
#### 1.2. Start the web-api service.
2019-08-16 15:50:07 -03:00
Invoke main.go directly or use `go build .`:
```bash
go run main.go
```
2019-08-16 15:50:07 -03:00
### 2. Initialize the MySQL database.
2019-08-16 15:50:07 -03:00
When `docker-compose up` is first ran, as discussed in the main readme,
the database is created but does not include any schema.
2019-08-16 15:50:07 -03:00
Thus, you need to use the project's schema tool to initialize the database.
To do this navigate to the `tools/schema` folder. Add the source to your env. Then run the `main.go` file for schema.
```bash
cd tools/schema/
source sample.env
go run main.go
```
2019-08-16 15:50:07 -03:00
Notice that if you run `main.go` again, you should see in the output than
all migrations `already ran` (as expected).
2019-08-16 15:50:07 -03:00
### 3. Open the Swagger UI.
Navigate your browser to [http://127.0.0.1:3001/docs](http://127.0.0.1:3001/docs).
2019-08-16 15:50:07 -03:00
### 4. Signup a new account.
Find the `signup` endpoint in the Swagger UI.
Click `Try it out`. Example data has been pre-populated to generate a valid POST request. Your can adjust the values
for the account and user objects accordingly.
```json
{
"account": {
"address1": "221 Tatitlek Ave",
"address2": "Box #1832",
"city": "Valdez",
"country": "USA",
"name": "Company 895ff280-5ed9-4b09-b7bc-86ab0f0951d4",
"region": "AK",
"timezone": "America/Anchorage",
"zipcode": "99686"
},
"user": {
"email": "90873f61-663e-43d1-8f0c-00415e73f650@example.com",
"name": "Gabi May",
"password": "SecretString",
"password_confirm": "SecretString"
}
}
```
**Note the user email and password from the above request will be used in the following steps.**
Click `Execute` and a response with status code 201 should have been returned.
```json
{
"account": {
"id": "baae6e0d-29ae-456f-9648-44c1e90ca8af",
"name": "Company 895ff280-5ed9-4b09-b7bc-86ab0f0951d4",
"address1": "221 Tatitlek Ave",
"address2": "Box #1832",
"city": "Valdez",
"region": "AK",
"country": "USA",
"zipcode": "99686",
"status": "active",
"timezone": "America/Anchorage",
"signup_user_id": {
"String": "bfdc5ca9-872c-4417-8030-e1b4962a107c",
"Valid": true
},
"billing_user_id": {
"String": "bfdc5ca9-872c-4417-8030-e1b4962a107c",
"Valid": true
},
"created_at": "2019-06-25T11:00:53.284Z",
"updated_at": "2019-06-25T11:00:53.284Z"
},
"user": {
"id": "bfdc5ca9-872c-4417-8030-e1b4962a107c",
"name": "Gabi May",
"email": "90873f61-663e-43d1-8f0c-00415e73f650@example.com",
"timezone": "America/Anchorage",
"created_at": "2019-06-25T11:00:53.284Z",
"updated_at": "2019-06-25T11:00:53.284Z"
}
}
```
If successful, data should be returned for code 201 for created.
Now you will now be able to use the email and password credentials to generate an auth token with the web-api service.
2019-08-16 15:50:07 -03:00
*Note: if Swagger `Try it out` response is `TypeError: Failed to fetch`, you probably accesed the swagger webpage via `http://localhost:3001/docs`. This would lead to a CORS error. You should open `http://127.0.0.1:3001/docs`*
2019-08-16 15:50:07 -03:00
### 5. Generate an Auth Token
An auth token is required for all other requests.
Near the top of the Swagger UI locate the button `Authorize` and click it.
Find the section `OAuth2Password (OAuth2, password)`
2019-08-16 15:50:07 -03:00
In `username` and `password` field fill the `email` value and `password` used in the signup process respectively.
Be careful that Swagger UI generates a random email in the signup process, it isn't the same as in section 4.
Click the button `Authorize` to generate a token that will be used by the Swagger UI for all future requests.
2019-08-16 15:50:07 -03:00
### 6. Test Auth Token
Now that the Swagger UI is authorized, try running endpoint using the oauth token.
2019-08-16 15:50:07 -03:00
Find the endpoint GET `/accounts/{id}` endpoint in the Swagger UI.
Click `Try it out` and enter the account ID from generated from signup (step 5).
Click `Execute`. The response should be of an Account.
2019-06-24 22:41:21 -08:00
2019-08-16 15:50:07 -03:00
### Authenticating Directly with Web API
If you want to authenticate directly with this web-api service and not via the Swagger UI, use the following steps.
2019-08-16 15:50:07 -03:00
#### 1. Authenticating
Before any authenticated requests can be sent you must acquire an auth token. Make a request using HTTP Basic auth with
your email and password to get an auth token.
```bash
curl --user "twin@example.com:SecretString" -X POST http://127.0.0.1:3001/v1/oauth/token
```
2019-08-16 15:50:07 -03:00
It is best to put the resulting token in an environment variable like `$TOKEN`. Notice that security tokens have an `expiry` date.
####2. Adding Token as Environment Variable
```bash
export TOKEN="COPY TOKEN STRING FROM LAST CALL"
```
####3. Authenticated Requests
To make authenticated requests put the token in the `Authorization` header with the `Bearer ` prefix.
```bash
curl -H "Authorization: Bearer ${TOKEN}" http://127.0.0.1:3001/v1/users
```
## Update Swagger API Documentation
2019-06-24 22:41:21 -08:00
Documentation is generated using [swag](https://github.com/geeks-accelerator/swag)
2019-06-24 22:41:21 -08:00
If you are developing this web-api service and you want your changes reflected in the API documentation, you will need
to download Swag and then run it each time you want the API documentation to be updated.
Download Swag with this command:
2019-06-24 22:41:21 -08:00
```bash
go get -u github.com/geeks-accelerator/swag/cmd/swag
2019-06-24 22:41:21 -08:00
```
Run `swag init` in the service's root folder which contains the main.go file. This will parse your comments and generate the required files (docs folder and docs/docs.go).
```bash
swag init
```
### Additional Swagger Annotations
2019-07-14 17:59:06 -08:00
Below are some additional example annotations that can be added to `main.go`
2019-07-14 17:59:06 -08:00
```go
// @title SaaS Example API
// @description This provides a public API...
// @termsOfService http://example.com/terms
// @contact.name API Support
// @contact.email support@geeksinthewoods.com
// @contact.url http://example.com/support
```
### Troubleshooting Swag
2019-06-24 22:41:21 -08:00
If you run into errors running `swag init` try the following:
#### cannot find package
Try to install the packages to your $GOPATH.
```bash
2019-08-05 19:49:30 -08:00
GO111MODULE=off go get -u github.com/leodido/go-urn
GO111MODULE=off go get -u github.com/lib/pq/oid
GO111MODULE=off go get -u github.com/lib/pq/scram
GO111MODULE=off go get -u github.com/tinylib/msgp/msgp
GO111MODULE=off go get -u gopkg.in/DataDog/dd-trace-go.v1/ddtrace
GO111MODULE=off go get -u golang.org/x/xerrors
2019-06-24 22:41:21 -08:00
```
#### error writing go.mod
Need to update pkg directory permissions.
Full error:
```bash
error writing go.mod: open /Users/leebrown/go/pkg/mod/github.com/lib/pq@v1.1.1/go.mod691440060.tmp: permission denied
```
Ensure the `pkg` directory used for go module cache has the correct permissions.
```bash
sudo chown -R $(whoami):staff ${HOME}/go/pkg
sudo chmod -R 755 ${HOME}/go/pkg
```