Skip to content

Latest commit

 

History

History
179 lines (110 loc) · 4.33 KB

README.md

File metadata and controls

179 lines (110 loc) · 4.33 KB

go-app-tmpl

A template for generic golang applications.

Nix Setup

Using Nix you can create a develop environment with all the tooling required to build this application.

This flake was initialized using a template.

nix flake init --template "github:DeterminateSystems/zero-to-nix#go-dev"

I expanded this to include the additional packages required to build.

You can enter the dev shell with the command nix develop.

Tooling

This section will cover all the moving parts of this demo. Each closes one gap for the fully declarative extensible build.

sqlc

sqlc is used to build out your database connection and query layers using just regular sql queries.

buf

buf is the new industry standard when it comes to protobuf generation. protoc is still used but buf is more declarative and extensible.

golang-migrate

golang-migrate migrate tool is used for database migrations. Its somewhat overkill for this but nice to show that it can be used along with sqlc and is easier to manage sql changes than traditional sql scripts.

Initialization

  1. Install the following tools.
  • sqlc cli

  • buf cli

  • golang-migrate cli

  • docker and docker-compose

  • make (optional)

  • grpcurl (optional for testing grpc server)

  1. Setup .env file - nothing special just to setup db creds.
export DB_HOST=localhost
export DB_PASSWORD=your_password
export DB_USER=your_username
export DB_NAME=golang_app

Note: the following are not needed if cloning this repo but ill cover for completeness. If you've cloned continue at build section.

  1. Initialize the go module.
go mod init github.com/alexander-bergeron/go-app-tmpl
  1. Initialize Docker files.

see file contents for further explanation

  • Dockerfile.server builds the golang server executable.

  • compose.yml is the docker-compose file for orchestration.

  1. Initialize migrations
migrate create -ext sql -dir migrations -seq init

This creates your migrations directory and creates two empty files, init.up and init.down.

-- init.up
-- Create the users table
CREATE TABLE users (
    user_id SERIAL PRIMARY KEY,
    username VARCHAR(50) NOT NULL UNIQUE,
    email VARCHAR(50) NOT NULL UNIQUE,
    first_name VARCHAR(100),
    last_name VARCHAR(100)
);

INSERT INTO users (username, email, first_name, last_name) VALUES
('Cow31337Killer', '[email protected]', 'Cow', 'Killer'),
('Durial321', '[email protected]', 'Durial', '321'),
('BigRedJapan', '[email protected]', 'BigRed', 'Japan');
-- init.down
DROP TABLE IF EXISTS users;
  1. Initialize sqlc.pgx.yaml.

see file contents for further explanation

  1. Initialize buf files.

see file contents for further explanation

Build Steps

  1. make build
  • This will generate the grpc protobuf code, the migrate directory with init files, and the sqlc.pgx.yaml code.

  • After build completes, manually update swagger.json with host and schemes.

  "info": {
    "title": "proto/user/v1/user.proto",
    "version": "version not set"
  },
  "host": "localhost:8080", # add this line
  "schemes": ["https"], # add this line
  "tags": [
    {
      "name": "UserService"
    }
  ],
  1. go mod tidy
  • Updates all your deps.
  1. make gen-certs
  • Creates cert files for tls.
  1. start docker daemon.

  2. make up

  • Runs docker compose.

Testing

  1. make test-get-grpc - test grpc client call

  2. make test-get-rest - test grpc-gateway

  3. make test-post-grpc - posts a new test user to the API

  4. Navigate to localhost:3000 for a webserver displaying all users with form input for a new user - built with htmx

  5. Navigate to localhost:8081 for a swagger page - Note: you will need to add the generated ca.crt to your browser for swagger to work.

TLS

Serving over TLS takes various steps.

  1. creating the certs - see Makefile

  2. Baking ca cert into swagger ui image

  3. updating browser to recognize your custom ca - import the ca.crt

TODO

  1. Determine how to set host automatically for swagger page rather than manual change after generation.

  2. Setup CORS so its not hard coded.

  3. setup static css and js

  4. test to see if air works in container - reference