Я использую веб-сервер labstack / echo и gofight для модульного тестирования. В процессе обучения go хотел бы знать, существует ли идиома go для доступа к состоянию вне (встроенной) структуры эха. Например:
type WebPlusDB struct {
web *echo.Echo
db *databaseInterface
}
func NewEngine() *echo.Echo {
e := echo.New()
e.GET("/hello", route_hello)
return e
}
func NewWebPlusDb() {
e := NewEngine()
db := database.New()
return WebPlusDB{e,db}
}
// for use in unit tests
func NewFakeEngine() *WebPlusDB {
e := NewEngine()
db := fakeDatabase.New()
return WebPlusDB{e,db}
}
func route_hello(c echo.Context) error {
log.Printf("Hello world\n")
// how do I access WebPlusDB.db from here?
return c.String(http.StatusOK, "hello world")
}
Тогда в коде теста я использую:
import (
"github.com/labstack/echo"
"github.com/appleboy/gofight"
"github.com/stretchr/testify/assert"
"testing"
)
func TestHelloWorld(t *testing.T) {
r := gofight.New()
r.GET("/hello").
Run(NewFakeEngine(), func(r gofight.HTTPResponse, rq gofight.HTTPRequest) {
assert.Equal(t, http.StatusOK, r.Code)
assert.Equal(t, "hello world", r.Body.String())
// test database access
})
}
Самое простое решение состоит в том, чтобы использовать глобальную переменную вместо встраивания echo в "WebPlusDB" и добавления туда состояния. Я хотел бы лучше инкапсуляции. Я думаю, что я должен использовать что-то вроде структуры WebPlusDB, а не echo.Echo плюс глобальное состояние. Возможно, это не имеет большого значения для модульного тестирования, но я хотел бы знать, что для более точной схемы правильного выполнения действий (в данном случае избегая глобальных).
Есть ли решение или это слабое место в дизайне эха?
У него есть точки расширения для промежуточного программного обеспечения, но серверная часть базы данных на самом деле не является промежуточным программным обеспечением, как определено здесь .
Примечание: я использую здесь базу данных для иллюстрации общего случая, но это может быть что угодно (на самом деле я использую amqp )
Похоже, вы можете расширить интерфейс context , но где он создан? Похоже, что он использует вид downcast :
e.GET("/", func(c echo.Context) error {
cc := c.(*CustomContext)
}
Я думаю (возможно, неправильно), что это разрешено только для интерфейса, и echo.Context.Echo () возвращает тип, а не интерфейс.