Adding a new post on golang unit tests
This commit is contained in:
parent
959212ab66
commit
92d9580e0a
|
@ -0,0 +1,366 @@
|
||||||
|
Title: Adventures in Golang (Tests)
|
||||||
|
Date: 2023-11-27 22:04
|
||||||
|
Category: Golang
|
||||||
|
Tags: Things I learned
|
||||||
|
Slug: golangtests
|
||||||
|
Authors: Okusanya David
|
||||||
|
Summary: About adding tests to a golang project
|
||||||
|
|
||||||
|
### OUTLINE
|
||||||
|
|
||||||
|
- [Problem statement](#golang-registry-tests-problem)
|
||||||
|
- [First Approach](#golang-registry-tests-approach-one)
|
||||||
|
- [Second Approach](#golang-registry-tests-approach-two)
|
||||||
|
|
||||||
|
### PROBLEM STATEMENT {#golang-registry-tests-problem}
|
||||||
|
|
||||||
|
So I had finished the first version of my code implementing a npm registry server in Go. This version has no tests. I have been testing it manually. I wanted to add filesystem locking and deploy the code through CI. So I needed to add automated tests(unit and integration) as I wanted complement my manual testing.
|
||||||
|
|
||||||
|
Here is a snippet of what I wanted to write a unit test for:
|
||||||
|
|
||||||
|
```golang
|
||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
|
||||||
|
"github.com/gorilla/mux"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
|
||||||
|
"gosimplenpm/internal/config"
|
||||||
|
"gosimplenpm/internal/storage"
|
||||||
|
)
|
||||||
|
|
||||||
|
func GetPackage(lg *logrus.Logger, cfg config.Config) http.HandlerFunc {
|
||||||
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
escapedName := mux.Vars(r)["name"]
|
||||||
|
packageName, _ := url.PathUnescape(escapedName)
|
||||||
|
lg.WithFields(logrus.Fields{
|
||||||
|
"function": "get-package",
|
||||||
|
}).Debugf("Package name => %s\n", packageName)
|
||||||
|
|
||||||
|
fileToServe, found, err := storage.GetIndexJsonFromStore(packageName, cfg.RepoDir, lg)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if !found {
|
||||||
|
ret := fmt.Sprintf("Package not found: %s", packageName)
|
||||||
|
http.Error(w, ret, http.StatusNotFound)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// serve file
|
||||||
|
http.ServeFile(w, r, fileToServe)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
So I wanted to test this function without also testing the `storage.GetIndexJsonFromStore` function. The `storage.GetIndexJsonFromStore` function makes calls to the filesystem. Well, coming from Javascript, I would use a mocking framework like Sinon.JS [^1] to mock out the `storage.GetIndexJsonFromStore` function without modifying the `GetPackage` handler.
|
||||||
|
|
||||||
|
### FIRST APPROACH {#golang-registry-tests-approach-one}
|
||||||
|
|
||||||
|
So using my limited knowledge of Go, I wrote an interface `Storage` and defined the method `GetIndexJsonFromStore` like below:
|
||||||
|
|
||||||
|
```golang
|
||||||
|
package storage
|
||||||
|
|
||||||
|
import (
|
||||||
|
"gosimplenpm/internal/serviceidos"
|
||||||
|
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Storage interface {
|
||||||
|
GetIndexJsonFromStore(string, string, *logrus.Logger) (string, bool, error)
|
||||||
|
// Other methods omitted for brevity
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
Then I can define a struct `MockFS` that would implement the method `GetIndexJsonFromStore`. This is because structs that implement all the methods of an interface are _objects_ of that interface.
|
||||||
|
|
||||||
|
```golang
|
||||||
|
|
||||||
|
type MockFs struct {
|
||||||
|
packageJson string
|
||||||
|
retrieved bool
|
||||||
|
err error
|
||||||
|
called bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockFs) SetError(err error) {
|
||||||
|
m.err = err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockFs) SetRetrieved(retrieved bool) {
|
||||||
|
m.retrieved = retrieved
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockFs) SetFileToServe(fileToServe string) {
|
||||||
|
m.packageJson = fileToServe
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockFs) GetIndexJsonFromStore(packageName string, registryPath string, lg *logrus.Logger) (string, bool, error) {
|
||||||
|
m.called = true
|
||||||
|
return m.packageJson, m.retrieved, m.err
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
I would then modify the previously defined `GetIndexJsonFromStore` method to allow for me to inject the `MockFs` dependency.
|
||||||
|
|
||||||
|
```golang
|
||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
|
||||||
|
"github.com/gorilla/mux"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
|
||||||
|
"gosimplenpm/internal/config"
|
||||||
|
"gosimplenpm/internal/storage"
|
||||||
|
)
|
||||||
|
|
||||||
|
func GetPackage(lg *logrus.Logger, cfg config.Config, stg storage.Storage) http.HandlerFunc {
|
||||||
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
escapedName := mux.Vars(r)["name"]
|
||||||
|
packageName, _ := url.PathUnescape(escapedName)
|
||||||
|
lg.WithFields(logrus.Fields{
|
||||||
|
"function": "get-package",
|
||||||
|
}).Debugf("Package name => %s\n", packageName)
|
||||||
|
|
||||||
|
fileToServe, found, err := stg.GetIndexJsonFromStore(packageName, cfg.RepoDir, lg)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if !found {
|
||||||
|
ret := fmt.Sprintf("Package not found: %s", packageName)
|
||||||
|
http.Error(w, ret, http.StatusNotFound)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// serve file
|
||||||
|
http.ServeFile(w, r, fileToServe)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
A sample test would be something like this
|
||||||
|
|
||||||
|
```golang
|
||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"gosimplenpm/internal/config"
|
||||||
|
"gosimplenpm/internal/storage"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"os"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/gorilla/mux"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGet(t *testing.T) {
|
||||||
|
t.Run("return `Not Found` error if package is not found", func(t *testing.T) {
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/{name}", nil)
|
||||||
|
wrt := httptest.NewRecorder()
|
||||||
|
|
||||||
|
log := &logrus.Logger{
|
||||||
|
Out: os.Stdout,
|
||||||
|
Formatter: &logrus.TextFormatter{
|
||||||
|
FullTimestamp: true,
|
||||||
|
TimestampFormat: "2009-01-02 15:15:15",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
cfg := config.Config{
|
||||||
|
RepoDir: "",
|
||||||
|
}
|
||||||
|
|
||||||
|
mfs := &storage.MockFs{}
|
||||||
|
mfs.SetRetrieved(false)
|
||||||
|
|
||||||
|
|
||||||
|
//Hack to try to fake gorilla/mux vars
|
||||||
|
vars := map[string]string{
|
||||||
|
"name": "test-package",
|
||||||
|
}
|
||||||
|
|
||||||
|
req = mux.SetURLVars(req, vars)
|
||||||
|
|
||||||
|
GetPackage(log, cfg, mfs)(wrt, req)
|
||||||
|
|
||||||
|
rs := wrt.Result()
|
||||||
|
|
||||||
|
assert.Equal(t, rs.StatusCode, http.StatusNotFound)
|
||||||
|
|
||||||
|
defer rs.Body.Close()
|
||||||
|
body, err := io.ReadAll(rs.Body)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
bytes.TrimSpace(body)
|
||||||
|
|
||||||
|
assert.Equal(t, string(body), "Package not found: test-package\n")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
This approach fails if I wanted to test another function `stg.GetOtherJsonFromNet` of the interface like below:
|
||||||
|
|
||||||
|
```golang
|
||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
|
||||||
|
"github.com/gorilla/mux"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
|
||||||
|
"gosimplenpm/internal/config"
|
||||||
|
"gosimplenpm/internal/storage"
|
||||||
|
)
|
||||||
|
|
||||||
|
func GetPackage(lg *logrus.Logger, cfg config.Config, stg storage.Storage) http.HandlerFunc {
|
||||||
|
return func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
escapedName := mux.Vars(r)["name"]
|
||||||
|
packageName, _ := url.PathUnescape(escapedName)
|
||||||
|
lg.WithFields(logrus.Fields{
|
||||||
|
"function": "get-package",
|
||||||
|
}).Debugf("Package name => %s\n", packageName)
|
||||||
|
|
||||||
|
fileToServe, found, err := stg.GetIndexJsonFromStore(packageName, cfg.RepoDir, lg)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !found {
|
||||||
|
fileToServe, err = stg.GetOtherJsonFromNet(packageName, lg)
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//...code omitted for brevity
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
### SECOND APPROACH {#golang-registry-tests-approach-two}
|
||||||
|
|
||||||
|
To handle this new scenario, that is, to test the other function `GetOtherJsonFromNet` as well as `GetIndexJsonFromStore`, I would modify the struct and its implementations to something like this:
|
||||||
|
|
||||||
|
```golang
|
||||||
|
|
||||||
|
type MockFs struct {
|
||||||
|
GetIndexJsonFromStoreFunc func(packageName string, registryPath string, lg *logrus.Logger) (string, bool, error)
|
||||||
|
GetOtherJsonFromNetFunc func(packageName string, lg *logrus.Logger) (string, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockFs) GetIndexJsonFromStore(packageName string, registryPath string, lg *logrus.Logger) (string, bool, error) {
|
||||||
|
return m.GetIndexJsonFromStoreFunc(packageName, registryPath, lg)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockFs) GetOtherJsonFromNet(packageName string, lg *logrus.Logger) (string, error) {
|
||||||
|
return m.GetOtherJsonFromNetFunc(packageName, lg)
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
Basically since I cannot reassign a function to an already defined struct (as I could in Javascript with objects), I would instead define function variables as part of the `MockFS` struct. That way, I can redefine those function variables for each test I want to run. An example of a test that uses this is shown below:
|
||||||
|
|
||||||
|
```golang
|
||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"gosimplenpm/internal/config"
|
||||||
|
"gosimplenpm/internal/storage"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"os"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/gorilla/mux"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestGet(t *testing.T) {
|
||||||
|
t.Run("return `Internal Server` error if package cannot be retrieved either from disk or from the internet", func(t *testing.T) {
|
||||||
|
req := httptest.NewRequest(http.MethodGet, "/{name}", nil)
|
||||||
|
wrt := httptest.NewRecorder()
|
||||||
|
|
||||||
|
log := &logrus.Logger{
|
||||||
|
Out: os.Stdout,
|
||||||
|
Formatter: &logrus.TextFormatter{
|
||||||
|
FullTimestamp: true,
|
||||||
|
TimestampFormat: "2009-01-02 15:15:15",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
cfg := config.Config{
|
||||||
|
RepoDir: "",
|
||||||
|
}
|
||||||
|
|
||||||
|
mfs := &storage.MockFs{}
|
||||||
|
mfs.GetIndexJsonFromStore = func (packageName string, registryPath string, lg *logrus.Logger) (string, bool, error) {
|
||||||
|
return "", false, fmt.Errorf("Failure to retrieve disk")
|
||||||
|
}
|
||||||
|
mfs.GetOtherJsonFromNet = func (packageName string, lg *logrus.Logger) (string, error) {
|
||||||
|
return "", false, fmt.Errorf("Failure to retrieve from internet")
|
||||||
|
}
|
||||||
|
|
||||||
|
//Hack to try to fake gorilla/mux vars
|
||||||
|
vars := map[string]string{
|
||||||
|
"name": "test-package",
|
||||||
|
}
|
||||||
|
|
||||||
|
req = mux.SetURLVars(req, vars)
|
||||||
|
|
||||||
|
GetPackage(log, cfg, mfs)(wrt, req)
|
||||||
|
|
||||||
|
rs := wrt.Result()
|
||||||
|
|
||||||
|
assert.Equal(t, rs.StatusCode, http.StatusInternalServerErro)
|
||||||
|
|
||||||
|
defer rs.Body.Close()
|
||||||
|
body, err := io.ReadAll(rs.Body)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
bytes.TrimSpace(body)
|
||||||
|
|
||||||
|
assert.Equal(t, string(body), "Failure to retrieve from internet\n")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
This approach avoids the usage of struct-level variables, making it easier to easily fine-tune your tests.
|
||||||
|
|
||||||
|
Now, I had already moved all my tests to this approach before I found out about GoMock[^2]. In the future, I may move to GoMock[^2] instead.
|
||||||
|
|
||||||
|
[^1]: [Sinon.JS](https://sinonjs.org/)
|
||||||
|
[^2]: [GoMock](https://github.com/golang/mock)
|
Loading…
Reference in New Issue