This is my first fullstack Go project. Built with Go version 1.19
This project is a simple hotel booking and reservation project with key features like;
- Authentication
- Create and manage rooms
- Showcase the rooms
- Check for rooms availability
- Allow booking of one room
- Notify guest and owner
- Have a backend for management
- Review existing bookings
- Show a calendar of bookings
- Change or cancel a booking
- Todo
- url - https://go-booking.onrender.com
- username/email: atu@prosper.com
- password: password
- Alex Edwards SCS package for managing sessions
- Chi router
- Justinas nosurf
- JackC PGX pgx is a pure Go driver and toolkit for PostgreSQL.
- Go Simple Mail Used for sending mails.
- Simple DataTable Used for tables.
- Buffalo Soda Used for tables.
- Create your own go mod file and delete the one used here, run the following command
go mod init your-project-name
- your-project-name is usually your github link and the name of your project, example "github.com/prosper74/go-project". This is not a must, but a recommendation.
- Change the name of every import to your current go mod name. Example, open the main.go file, in the
required imports
section, replace these "github.com/atuprosper/go-project/pkg/config" to "github.com/atuprosper/your-project-name/pkg/config". Go through all files and make this replacement - After all the necessary changes, run the app
go run cmd/web/*.go
this will install all the third party packages and run the server on the selected port. - Create your postgres db
- Setup the flags in main.go file -
cmd/web/main.go
- Setup the flags in run.sh file
- Do not use the rub.bat file as it encounters errors sometimes from windows. run.sh will work for both windows and linux
- Setup the .env file, rename the
.env.example
to.env
. Create your sendinBlue account and add the api - Setup the
database.yml
, rename thedatabase.yml.example
todatabase.yml
. This will enable you to runsoda migrate
-
Manual:
go run cmd/web/main.go cmd/web/middleware.go cmd/web/routes.go cmd/web/sendMail.go
-
Batch:
On Windows - create arun.bat
file in the root directory of the project and paste the below codego build -o bookings cmd/web/*.go ./bookings.exe
Then run
run.bat
in the terminalOn linux - create a
run.sh
file in the root directory of the project and paste the below code#!/bin/bash go build -o booking cmd/web/*.go ./bookings
Run
chmod +x run.sh
then run./run.sh
in the terminal
- To output test in hmtl format run
go test -coverprofile=coverage.out && go tool cover -html=coverage.out
- To know the percentage coverage run
go test -cover
- Run test for the entire project
go test -v ./...
- install
soda
, rungo install github.com/gobuffalo/pop/v6/soda@latest
soda g config
to generate a database.yml file in the current directory for a PostgreSQL database. Then setup your database- run
soda generate fizz migration-name
in the terminal, to create the migration folder and files. Run this code to create migration files for each table - We can also run sql version
soda generate sql migration-name
- The up file is used to create or update while the down file is used to delete or downgrade
- run
soda migrate
to run the migration files - run
soda migrate down
to run the down migration files - run
soda reset
to drop everything in the database and create the migration again - Read the documentation to know more about Buffalo or the Fizz Github page
This is where we create and configure our session
-
The seesion variable
var session *scs.SessionManager
here and point (*) it to the scs package. This will make the session variable available to other files in this project. -
Here
session = scs.New()
we assign a new scs session to the sessions variable. -
session.Lifetime = 24 * time.Hour
states how long will the sessions last.time.Hour
is a Go built in package -
session.Cookie.Persist = true
Here we store the session in a cookie. We can also use other databases like redis to store our session. This will make the session persist for this time duration even if the browser window is closed and openned again. If its set to false, the session ends when the brower is closed -
To set the strictness of the session, we use
session.Cookie.SameSite = http.SameSiteLaxMode
-
To make sure the connection is encrypted (i.e https), we use
session.Cookie.Secure = app.InProduction
. In production, this should be true. In development, set it to false. Note thatapp.InProduction
is set in the site wideconfig.go
file -
Finally, we make our session available site wide by passing it to
app.Session = session
This where we handle all our pages and templates. This is where we use our scs package and all it's features. We want to collect the IP address of our users and store them in the session, then send them to be used in our HTML templates
func (m *Repository) Home(w http.ResponseWriter, r *http.Request)
A function with a 'reciever' m, of type 'Repository'. This will give our handler function access to everything in the config filevar Repo *Repository
Creating a Repository pattern. This variable is the repository used by the handlers- Get the IP address of user using the r parameter, which holds the
*http.Request
Go built in package -remoteIP := r.RemoteAddr
- Store the IP in the site wide config which is available via the paramter 'm'. It takes in three parameters. The context, the name (which can be anything), and the value -
m.App.Session.Put(r.Context(), "remote_ip", remoteIP)
- Get the remote IP from the session -
getRemoteIP := m.App.Session.GetString(r.Context(), "remote_ip")
- Store the retrieved IP in the
stringMap
which is available in our HTML template -stringMap["remote_ip"] = getRemoteIP
This is where we handle routes, multiplexers and middlewares
mux := chi.NewRouter()
Creates a multiplexer Make the templates recognize the static folder to use its contents (eg images)- Create a file server for the static directory
fileServer := http.FileServer(http.Dir("./static/"))
- Create a multiplexer to handle the fileServer. "http.StripePrefix" takes the url and modifies it to something it can handle.
mux.Handle("/static/*", http.StripPrefix("/static", fileServer))
- 'Recoverer' middleware from chi package. It helps for panic control
mux.Use(middleware.Recoverer)
template.FuncMap{}
A FuncMap is a map of functions that we can use in our templateNewTemplates
sets the config for the template packageAddDefaultData
This function adds default data to every templates by taking the specific data sent from the render function and attach this function to it
We can write our own middleware for chi
package in this file.
next
is commonly used as the parameter for custome middlewareNoSurf
middleware adds CSRF protection to all POST request -func NoSurf(next http.Handler) http.Handler
SessionLoad
middleware makes our server to be 'state' aware, in order to keep our session in statefunc (next http.Handler) http.Handler
LoadAndSave
provides middleware which automatically loads and saves session data for the current request, and communicates the session token to and from the client in a cookie.
This file should be made accessible by other packages, every package in this project can import it and use it. But this file should not import any package, it will only use the standard built-in Go packages. This will help us avoid an error called "import cycle"
AppConfig
struct holds the application configurationsInfoLog
allows you to create a log file and store informations in it
We create this seperate file to avoid Go import cycle not allowed error
TemplateData struct
is a struct that holds data sent from handlers to templatesinterface{}
type is used when the type is not known- Cross Site Request Forgery Token (CSRFToken) - is a security token to handle forms
- To output test in hmtl format run
go test -coverprofile=coverage.out && go tool cover -html=coverage.out
- To know the percentage coverage run
go test -cover