Skip to main content

Simple Web App backed by a database

This guide assumes you've gone through our getting started and set up Rig on Docker locally. We will create a simple webserver backed by a MongoDB database which implements the backend to a cake-rating app.

mkdir database
cd database

and follow the instructions in the guide to setup the Golang SDK. You can also clone this example here

Simple database examples setup

Our project will contain a main.go, mongo.go, go.mod and go.sum files with main.go powering the application. We will also have a Dockerfile so we can make a Docker image and deploy it as a Rig capsule. The file structure will be

database-demo
├── Dockerfile
├── main.go
├── mongo.go
├── go.mod
├── go.sum

Run

go get go.mongodb.org/mongo-driver

to get the go dependencies. Our Dockerfile will contain

Dockerfile
FROM golang:1.20

WORKDIR /usr/src/app

COPY go.mod go.sum ./
RUN go mod download && go mod verify

COPY . .

CMD ["go", "run", "main.go", "mongo.go"]

main.go will for now just contain an almost empty main function initializing a rig.Client and a test just to see that we can reach the Rig backend

main.go
var client rig.Client

func main() {
client = rig.NewClient()
if _, err := client.Database().List(context.Background(), connect.NewRequest(&database.ListRequest{})); err != nil {
log.Fatal(err)
}
fmt.Println("Successfully connected to Rig!")
}

Deploying as a Rig capsule

We will run the application by deploying it as a Rig capsule running locally. This also allows us to easier integrate with the Rig authorization workflow. Start by making a new capsule

rig capsule create database-demo

Then make a Docker image of the TODO demo

docker build -t database-demo .

This we will deploy to our new database-demo capsule

rig capsule create-build database-demo --image database-demo --deploy

Now we should have database-demo running locally which we can verify by running docker ps

> docker ps
CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAMES
0294f8e4d7bc database-demo:latest "go run ./main.go" 2 seconds ago Up 1 second database-demo-instance-0

The rig.Client expects credentials to be present in the environment variables RIG_CLIENT_ID and RIG_CLIENT_SECRET. We can automatically inject these in our capsule by running

rig capsule config database-demo --auto-add-service-account

If things work you should be able to inspect the logs of the capsule

rig capsule logs database-demo

which hopefully shows

Successfully connected to Rig!

Whenever you want to re-deploy the capsule with new code changes, you can update the Docker image and then deploy it

docker build -t database-demo .
rig capsule create-build database-demo --image database-demo --deploy

Setting up a new MongoDB managed by Rig

We will use Rig to manage our database and do the work through Rig's CLI. Run

rig database create --name our_db --type mongo

This will create a new MongoDB database. Currently, that is the only database we natively support, but more will come. Running

rig database list

should output

+---------+--------+------------+
| DBS (1) | NAME | TYPE |
+---------+--------+------------+
| 1 | our_db | TYPE_MONGO |
+---------+--------+------------+

which confirms we have successfully created one. Although it exists, we don't have any credentials on it which means we can't access it yet. Therefore we will add some credentials

rig database create-credentials

and write our_db once it prompts you for a DB Identiifer. This should output a new credential e.g.

created credential - clientID: rig_a9..., secret: secret_fe...

Copy the clientID and secret somewhere you can find them again, as Rig won't have access to the secret again.

Implementing a backend using the database

We need some static configuration for the database we want to connect to, specifically the name and the credentials created previously.

main.go
type dbConfig struct {
dbname string
username string
password string
}

var mongoConfig = dbConfig{
dbname: os.Getenv("DATABASE_NAME"),
username: os.Getenv("DATABASE_CLIENT_ID"),
password: os.Getenv("DATABASE_CLIENT_SECRET"),
}

We will read the database name and credentials from environment variables, and we can configure these environment variables within the Rig capsule that deploys our application. We will get back to that a bit later. With that let's try to connect to our DB

main.go
func main() {
client = rig.NewClient()
ctx := context.Background()

// When Rig creates a new database it associates to it a UUID. We need this UUID
dbResponse, err := client.Database().GetByName(ctx, connect.NewRequest(&database.GetByNameRequest{
Name: mongoConfig.dbname,
}))
if err != nil {
return fmt.Errorf("no database found: %q", err)
}
dbUUID := dbResponse.Msg.Database.Id

// GetEndpoint returns the URI needed to open a connection to the database given by the DatabaseId
// and username/password supplied by the ClientID and ClientSecret
endpointResponse, err := client.Database().GetEndpoint(ctx, connect.NewRequest(&database.GetEndpointRequest{
DatabaseId: dbUUID,
ClientID: mongoConfig.username,
ClientSecret: mongoConfig.password,
}))
if err != nil {
log.Fatal(fmt.Errorf("failed to get endpoint: %q", err))
}
uriToDatabase := endpointResponse.Msg.Endpoint
fmt.Printf("uri: %s\n", uriToDatabase)

// Now that we have an URI to the database, we can establish a connection
mongoClient, err := mongo.Connect(ctx, options.Client().ApplyURI(uriToDatabase))
if err != nil {
return nil, fmt.Errorf("failed to connect to mongo: %q", err)
}
// Although we now have a connection to the database, we need an explicit handle to it
// This requires us to pass in the name which Rig gave to MongoDB when creating it.
// This name differs from the `our_db` we called it when constructing it through the Rig CLI
db := mongoClient.Database(endpointResponse.Msg.DatabaseName)
}

This should give us a valid handle to a *mongo.Database object interfacing with our database. If you re-deploy the capsule with just these changes, the environment variables we read into the dbConfig will not be set. To set them in the capsule, go to your local Rig dashboard (localhost:4747) and under the Settings tab of your database-demo capsule. Here you can set environment variables as shown below image

It's time to use our database connection! Our cake-rating app will have a backend with the following:

  • A collection of Images each having an ELO rating and a URL to an image of the cake. These will be stored in our database.
  • An endpoint addImage which adds a new image to the collection with a URL an image.
  • An endpoint getPair which returns a uniformly random pair of two Images
  • An endpoint vote which votes on which of the two cakes (Images) is best. This will update the ELO rating of those two Images.
  • An endpoint listImages which dumps the collection of all images

Each of these endpoints is served by a unique database query. We'll encapsulate these queries in a Repository interface so we can easily add support for other databases once we support it

main.go
type Repository interface {
// The Setup function we use to initially construct the `images` table if it doesn't exist
Setup(ctx context.Context) error
AddImage(ctx context.Context, url string) (image, error)
GetPair(ctx context.Context) ([2]image, error)
Vote(ctx context.Context, winnerUUID, loserUUID string) error
ListImages(ctx context.Context) ([]image, error)
}

type image struct {
Id string `json:"id"`
Url string `json:"url"`
Elo float64 `json:"elo"`
}

Make the file mongo.go where we'll implement the interface and move the connection logic. For now, we'll just have empty implementations of the interface functions.

mongo.go
package main

func newMongoRepository(ctx context.Context, uri string, our_db string) (Repository, error) {
mongoClient, err := mongo.Connect(ctx, options.Client().ApplyURI(uri))
if err != nil {
return nil, fmt.Errorf("failed to connect to mongo: %q", err)
}

return &mongoRepository{
db: mongoClient.Database(our_db),
}, nil
}

type mongoRepository struct {
db *mongo.Database
}

func (m *mongoRepository) Setup(ctx context.Context) error {
return nil
}

func (m *mongoRepository) AddImage(ctx context.Context, url string) (image, error) {
return image{}, nil
}

func (m *mongoRepository) GetPair(ctx context.Context) ([2]image, error) {
var empty [2]image
return empty, nil
}

func (m *mongoRepository) Vote(ctx context.Context, winnerUUID, loserUUID string) error {
return nil
}

func (m *mongoRepository) ListImages(ctx context.Context) ([]image, error) {
return nil, nil
}

We also need a couple of changes to main.go to accommodate this new Repository setup

main.go
var client rig.Client
var repo Repository

func main() {
client = rig.NewClient()
ctx := context.Background()

// When Rig creates a new database it associates to it a UUID. We need this UUID
dbResponse, err := client.Database().GetByName(ctx, connect.NewRequest(&database.GetByNameRequest{
Name: mongoConfig.our_db,
}))
if err != nil {
return fmt.Errorf("no database found: %q", err)
}
dbUUID := dbResponse.Msg.Database.Id

// GetEndpoint returns the URI needed to open a connection to the database given by the DatabaseId
// and username/password supplied by the ClientID and ClientSecret
endpointResponse, err := client.Database().GetEndpoint(ctx, connect.NewRequest(&database.GetEndpointRequest{
DatabaseId: dbUUID,
ClientID: mongoConfig.username,
ClientSecret: mongoConfig.password,
}))
if err != nil {
log.Fatal(fmt.Errorf("failed to get endpoint: %q", err))
}
uriToDatabase := endpointResponse.Msg.Endpoint
fmt.Printf("uri: %s\n", uriToDatabase)

repo, err = newMongoRepository(ctx, uriToDatabase, endpointResponse.Msg.DatabaseName)
if err != nil {
log.Fatal("failed to make repository: %q", err)
}

if err := repo.Setup(ctx); err != nil {
log.Fatal("failed to setup database: %q", err)
}
}

The next step is to implement the Setup function to create an images collection if it doesn't already exist

mongo.go
func (m *mongoRepository) Setup(ctx context.Context) error {
collections, err := m.db.ListCollectionNames(ctx, bson.D{})
if err != nil {
return err
}
if slices.Contains(collections, "images") {
return nil
}
return m.db.CreateCollection(ctx, "images")
}

Adding an Image

If you have set up the database correctly and passed in the right credentials, running go run *.go shouldn't produce any errors. Next, we'll add functionality to add a new Image to our collection.

mongo.go
func (m *mongoRepository) AddImage(ctx context.Context, url string) (image, error) {
collection := m.db.Collection("images")
response, err := collection.InsertOne(ctx, bson.D{
{Key: "url", Value: url},
{Key: "elo", Value: 1000}, // We choose a fixed base elo of 1,000
})
if err != nil {
return image{}, err
}
oid := response.InsertedID.(primitive.ObjectID)
return image{
Id: oid.String(),
Url: url,
Elo: 1000,
}, nil
}

As I'm envisioning this backend to function as a webserver I'll access this AddImage function from a request. Before doing that I need a helper function that is purely for ergonomics relating to handling HTTP requests

main.go
// requestWrapper wraps a http request handler which uses a Context and returns an error to a function with the standard header
func requestWrapper(handler func(ctx context.Context, w http.ResponseWriter, r *http.Request) error) func(w http.ResponseWriter, r *http.Request) {
return func(w http.ResponseWriter, r *http.Request) {
ctx := context.Background()
if err := handler(ctx, w, r); err != nil {
fmt.Printf("error: %s\n", err.Error())
w.WriteHeader(http.StatusInternalServerError)
w.Write([]byte(err.Error()))
}
}
}

Time to make main.go a web server!

main.go
func main() {
// The rest of the function...

runServer()
}

func runServer() error {
http.HandleFunc("/addImage", requestWrapper(addImage))

err := http.ListenAndServe(":3333", nil)
return err
}

func addImage(ctx context.Context, w http.ResponseWriter, r *http.Request) error {
imgURL := r.URL.Query().Get("imgurl")
img, err := repo.AddImage(ctx, imgURL)
if err != nil {
return err
}
w.Write([]byte(img.Id))
return nil
}

The server listens on port 3333 and for now, has a single endpoint /addImage which takes a query parameter imgurl. One technical thing about passing URLs as a query parameter is that we first need to convert it to an encoding valid for an HTTP request. I usually use this. As an example, I've chosen this lovely image. Starting the server by running go run *.go I can add the image to our database by going to this address

http://localhost:3333/addImage?imgurl=https%3A%2F%2Fimages.pexels.com%2Fphotos%2F3323686%2Fpexels-photo-3323686.jpeg%3Fauto%3Dcompress%26cs%3Dtinysrgb%26w%3D1600

If everything works as it should the browser should print the ID of the entry in the MongoDB, e.g.

ObjectID("64e7473a2e55308c8e677e45")

At this point, it is just a matter of implementing the remaining endpoints as they correspond to functions in the Repository interface. For those interested, the rest of this article is my implementation.

List Images

mongo.go
func (m *mongoRepository) ListImages(ctx context.Context) ([]image, error) {
collection := m.db.Collection("images")
cur, err := collection.Find(ctx, bson.D{})
if err != nil {
return nil, err
}

var mongoImages []mongoImage
if err := cur.All(ctx, &mongoImages); err != nil {
return nil, err
}

var images []image
for _, img := range mongoImages {
images = append(images, img.toImage())
}
return images, nil
}

type mongoImage struct {
Id primitive.ObjectID `json:"_id" bson:"_id"`
Url string `json:"url" bson:"url"`
Elo float64 `json:"elo" bson:"elo"`
}

func (m mongoImage) toImage() image {
return image{
Id: m.Id.String(),
Url: m.Url,
Elo: m.Elo,
}
}
main.go
func runServer() error {
http.HandleFunc("/addImage", requestWrapper(addImage))
http.HandleFunc("/listImages", requestWrapper(listImages))

err := http.ListenAndServe(":3333", nil)
return err
}

func listImages(ctx context.Context, w http.ResponseWriter, r *http.Request) error {
images, err := repo.ListImages(ctx)
if err != nil {
return err
}
jsonBytes, err := json.Marshal(&images)
if err != nil {
return err
}
w.Write(jsonBytes)

return nil
}

After adding a few images, going to http://localhost:3333/listImages should display something like

[
{
"id": "ObjectID("64e746fd2e55308c8e677e42")",
"url": "https://preppykitchen.com/wp-content/uploads/2018/04/Funfetti-cake-recipe-new.jpg",
"elo": 1000
},
{
"id": "ObjectID("64e7470f2e55308c8e677e43")",
"url": "https://preppykitchen.com/wp-content/uploads/2018/04/Funfetti-cake-recipe-new.jpg",
"elo": 1000
},
{
"id": "ObjectID("64e748cc2e55308c8e677e47")",
"url": "https://images.pexels.com/photos/3323686/pexels-photo-3323686.jpeg?auto=compress\u0026cs=tinysrgb\u0026w=1600",
"elo": 1000
}
]

Get An Image Pair

mongo.go
func (m *mongoRepository) GetPair(ctx context.Context) ([2]image, error) {
var empty [2]image

collection := m.db.Collection("images")
cursor, err := collection.Aggregate(ctx, mongo.Pipeline{
{
{
Key: "$sample",
Value: bson.D{{
Key: "size",
Value: 2,
}},
},
},
})
if err != nil {
return empty, fmt.Errorf("failed to sample 2: %q", err)
}

var mongoImages [2]mongoImage
if err := cursor.All(ctx, &mongoImages); err != nil {
return empty, err
}

return [2]image{
mongoImages[0].toImage(),
mongoImages[1].toImage(),
}, nil
}
main.go
func runServer() error {
http.HandleFunc("/addImage", requestWrapper(addImage))
http.HandleFunc("/listImages", requestWrapper(listImages))
http.HandleFunc("/pair", requestWrapper(pair))

err := http.ListenAndServe(":3333", nil)
return err
}

func pair(ctx context.Context, w http.ResponseWriter, r *http.Request) error {
images, err := repo.GetPair(ctx)
if err != nil {
return err
}
jsonBytes, err := json.Marshal(&images)
if err != nil {
return err
}
w.Write(jsonBytes)
return nil
}

Vote on an image pair

mongo.go
func (m *mongoRepository) Vote(ctx context.Context, winnerUUID, loserUUID string) error {
winnerID, err := primitive.ObjectIDFromHex(winnerUUID)
if err != nil {
return err
}
loserID, err := primitive.ObjectIDFromHex(loserUUID)
if err != nil {
return err
}

collection := m.db.Collection("images")
var winnerImg image
if err := collection.FindOne(ctx, bson.M{"_id": winnerID}).Decode(&winnerImg); err != nil {
return err
}

var loserImg image
if err := collection.FindOne(ctx, bson.M{"_id": loserID}).Decode(&loserImg); err != nil {
return err
}

// Implement proper ELO update calculation
winnerImg.Elo += 100
loserImg.Elo -= 100
if _, err := collection.UpdateByID(ctx, winnerID, bson.M{"$set": bson.M{"elo": winnerImg.Elo}}); err != nil {
return err
}
if _, err := collection.UpdateByID(ctx, loserID, bson.M{"$set": bson.M{"elo": loserImg.Elo}}); err != nil {
return err
}

return nil
}

The vote endpoint assumes the winner and loser of a vote are given by string UUIDs in the parameters winner and loser.

main.go
func runServer() error {
// The rest of the function...
http.HandleFunc("/vote", requestWrapper(vote))

err := http.ListenAndServe(":3333", nil)
return err
}

func vote(ctx context.Context, w http.ResponseWriter, r *http.Request) error {
winner := r.URL.Query().Get("winner")
loser := r.URL.Query().Get("loser")
return repo.Vote(ctx, winner, loser)
}

With this code you should have a very basic backend with a database managed by Rig which can implement a rating app, in this case for cakes 🍰


Was this page helpful?