fix: missing service tests #181
All checks were successful
Build Docker Image / Build-Docker-Image (push) Successful in 50s
Build and Push Docker Image / Build-And-Push-Docker-Image (push) Successful in 2m17s

This commit was merged in pull request #202.
This commit is contained in:
2024-10-06 10:05:00 +02:00
parent 4dfd29eac1
commit 0fab1e1f2e
6 changed files with 95 additions and 23 deletions

View File

@@ -6,6 +6,7 @@ packages:
interfaces: interfaces:
RandomGenerator: RandomGenerator:
Clock: Clock:
MailService:
me-fit/db: me-fit/db:
interfaces: interfaces:
DbAuth: DbAuth:

View File

@@ -145,7 +145,7 @@ func (handler HandlerAuthImpl) handleSignUp() http.HandlerFunc {
return nil, err return nil, err
} }
go handler.service.SendVerificationMail(user) go handler.service.SendVerificationMail(user.Id, user.Email)
return nil, nil return nil, nil
}) })

View File

@@ -18,7 +18,8 @@ func GetHandler(d *sql.DB, serverSettings *types.ServerSettings) http.Handler {
randomGenerator := service.NewRandomGeneratorImpl() randomGenerator := service.NewRandomGeneratorImpl()
clock := service.NewClockImpl() clock := service.NewClockImpl()
dbAuth := db.NewDbAuthSqlite(d) dbAuth := db.NewDbAuthSqlite(d)
serviceAuth := service.NewServiceAuthImpl(dbAuth, randomGenerator, clock, serverSettings) mailService := service.NewMailServiceImpl(serverSettings)
serviceAuth := service.NewServiceAuthImpl(dbAuth, randomGenerator, clock, mailService, serverSettings)
handlerAuth := NewHandlerAuth(d, serviceAuth, serverSettings) handlerAuth := NewHandlerAuth(d, serviceAuth, serverSettings)
// Serve static files (CSS, JS and images) // Serve static files (CSS, JS and images)

View File

@@ -47,24 +47,24 @@ func NewUser(user *db.User) *User {
type ServiceAuth interface { type ServiceAuth interface {
SignIn(email string, password string) (*User, error) SignIn(email string, password string) (*User, error)
SignUp(email string, password string) (*User, error) SignUp(email string, password string) (*User, error)
SendVerificationMail(user *User) SendVerificationMail(userId uuid.UUID, email string)
} }
type ServiceAuthImpl struct { type ServiceAuthImpl struct {
dbAuth db.DbAuth dbAuth db.DbAuth
randomGenerator RandomGenerator randomGenerator RandomGenerator
clock Clock clock Clock
serverSettings *types.ServerSettings
mailService MailService mailService MailService
serverSettings *types.ServerSettings
} }
func NewServiceAuthImpl(dbAuth db.DbAuth, randomGenerator RandomGenerator, clock Clock, serverSettings *types.ServerSettings) *ServiceAuthImpl { func NewServiceAuthImpl(dbAuth db.DbAuth, randomGenerator RandomGenerator, clock Clock, mailService MailService, serverSettings *types.ServerSettings) *ServiceAuthImpl {
return &ServiceAuthImpl{ return &ServiceAuthImpl{
dbAuth: dbAuth, dbAuth: dbAuth,
randomGenerator: randomGenerator, randomGenerator: randomGenerator,
clock: clock, clock: clock,
mailService: mailService,
serverSettings: serverSettings, serverSettings: serverSettings,
mailService: NewMailService(serverSettings),
} }
} }
@@ -123,10 +123,10 @@ func (service ServiceAuthImpl) SignUp(email string, password string) (*User, err
return NewUser(dbUser), nil return NewUser(dbUser), nil
} }
func (service ServiceAuthImpl) SendVerificationMail(user *User) { func (service ServiceAuthImpl) SendVerificationMail(userId uuid.UUID, email string) {
var token string var token string
token, err := service.dbAuth.GetEmailVerificationToken(user.Id) token, err := service.dbAuth.GetEmailVerificationToken(userId)
if err != nil { if err != nil {
return return
} }
@@ -137,7 +137,7 @@ func (service ServiceAuthImpl) SendVerificationMail(user *User) {
return return
} }
err = service.dbAuth.InsertEmailVerificationToken(user.Id, token) err = service.dbAuth.InsertEmailVerificationToken(userId, token)
if err != nil { if err != nil {
return return
} }
@@ -150,7 +150,7 @@ func (service ServiceAuthImpl) SendVerificationMail(user *User) {
return return
} }
service.mailService.SendMail(user.Email, "Welcome to ME-FIT", w.String()) service.mailService.SendMail(email, "Welcome to ME-FIT", w.String())
} }
// TODO // TODO
@@ -311,7 +311,7 @@ func HandleSignOutComp(db *sql.DB) http.HandlerFunc {
} }
func HandleDeleteAccountComp(db *sql.DB, serverSettings *types.ServerSettings) http.HandlerFunc { func HandleDeleteAccountComp(db *sql.DB, serverSettings *types.ServerSettings) http.HandlerFunc {
mailService := NewMailService(serverSettings) mailService := NewMailServiceImpl(serverSettings)
return func(w http.ResponseWriter, r *http.Request) { return func(w http.ResponseWriter, r *http.Request) {
user := utils.GetUserFromSession(db, r) user := utils.GetUserFromSession(db, r)
if user == nil { if user == nil {
@@ -508,7 +508,7 @@ func HandleActualResetPasswordComp(db *sql.DB) http.HandlerFunc {
} }
func HandleResetPasswordComp(db *sql.DB, serverSettings *types.ServerSettings) http.HandlerFunc { func HandleResetPasswordComp(db *sql.DB, serverSettings *types.ServerSettings) http.HandlerFunc {
mailService := NewMailService(serverSettings) mailService := NewMailServiceImpl(serverSettings)
return func(w http.ResponseWriter, r *http.Request) { return func(w http.ResponseWriter, r *http.Request) {
email := r.FormValue("email") email := r.FormValue("email")

View File

@@ -4,6 +4,7 @@ import (
"me-fit/db" "me-fit/db"
"me-fit/mocks" "me-fit/mocks"
"me-fit/types" "me-fit/types"
"strings"
"errors" "errors"
"testing" "testing"
@@ -11,6 +12,7 @@ import (
"github.com/google/uuid" "github.com/google/uuid"
"github.com/stretchr/testify/assert" "github.com/stretchr/testify/assert"
"github.com/stretchr/testify/mock"
) )
func TestSignIn(t *testing.T) { func TestSignIn(t *testing.T) {
@@ -35,8 +37,9 @@ func TestSignIn(t *testing.T) {
mockDbAuth.EXPECT().GetUser("test@test.de").Return(user, nil) mockDbAuth.EXPECT().GetUser("test@test.de").Return(user, nil)
mockRandom := mocks.NewMockRandomGenerator(t) mockRandom := mocks.NewMockRandomGenerator(t)
mockClock := mocks.NewMockClock(t) mockClock := mocks.NewMockClock(t)
mockMail := mocks.NewMockMailService(t)
underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, &types.ServerSettings{}) underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, mockMail, &types.ServerSettings{})
actualUser, err := underTest.SignIn(user.Email, "password") actualUser, err := underTest.SignIn(user.Email, "password")
assert.Nil(t, err) assert.Nil(t, err)
@@ -70,8 +73,9 @@ func TestSignIn(t *testing.T) {
mockDbAuth.EXPECT().GetUser(user.Email).Return(user, nil) mockDbAuth.EXPECT().GetUser(user.Email).Return(user, nil)
mockRandom := mocks.NewMockRandomGenerator(t) mockRandom := mocks.NewMockRandomGenerator(t)
mockClock := mocks.NewMockClock(t) mockClock := mocks.NewMockClock(t)
mockMail := mocks.NewMockMailService(t)
underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, &types.ServerSettings{}) underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, mockMail, &types.ServerSettings{})
_, err := underTest.SignIn("test@test.de", "wrong password") _, err := underTest.SignIn("test@test.de", "wrong password")
@@ -84,8 +88,9 @@ func TestSignIn(t *testing.T) {
mockDbAuth.EXPECT().GetUser("test").Return(nil, db.ErrUserNotFound) mockDbAuth.EXPECT().GetUser("test").Return(nil, db.ErrUserNotFound)
mockRandom := mocks.NewMockRandomGenerator(t) mockRandom := mocks.NewMockRandomGenerator(t)
mockClock := mocks.NewMockClock(t) mockClock := mocks.NewMockClock(t)
mockMail := mocks.NewMockMailService(t)
underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, &types.ServerSettings{}) underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, mockMail, &types.ServerSettings{})
_, err := underTest.SignIn("test", "test") _, err := underTest.SignIn("test", "test")
assert.Equal(t, ErrInvaidCredentials, err) assert.Equal(t, ErrInvaidCredentials, err)
@@ -97,8 +102,9 @@ func TestSignIn(t *testing.T) {
mockDbAuth.EXPECT().GetUser("test").Return(nil, errors.New("Some undefined error")) mockDbAuth.EXPECT().GetUser("test").Return(nil, errors.New("Some undefined error"))
mockRandom := mocks.NewMockRandomGenerator(t) mockRandom := mocks.NewMockRandomGenerator(t)
mockClock := mocks.NewMockClock(t) mockClock := mocks.NewMockClock(t)
mockMail := mocks.NewMockMailService(t)
underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, &types.ServerSettings{}) underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, mockMail, &types.ServerSettings{})
_, err := underTest.SignIn("test", "test") _, err := underTest.SignIn("test", "test")
@@ -114,8 +120,9 @@ func TestSignUp(t *testing.T) {
mockDbAuth := mocks.NewMockDbAuth(t) mockDbAuth := mocks.NewMockDbAuth(t)
mockRandom := mocks.NewMockRandomGenerator(t) mockRandom := mocks.NewMockRandomGenerator(t)
mockClock := mocks.NewMockClock(t) mockClock := mocks.NewMockClock(t)
mockMail := mocks.NewMockMailService(t)
underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, &types.ServerSettings{}) underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, mockMail, &types.ServerSettings{})
_, err := underTest.SignUp("invalid email address", "SomeStrongPassword123!") _, err := underTest.SignUp("invalid email address", "SomeStrongPassword123!")
@@ -127,8 +134,9 @@ func TestSignUp(t *testing.T) {
mockDbAuth := mocks.NewMockDbAuth(t) mockDbAuth := mocks.NewMockDbAuth(t)
mockRandom := mocks.NewMockRandomGenerator(t) mockRandom := mocks.NewMockRandomGenerator(t)
mockClock := mocks.NewMockClock(t) mockClock := mocks.NewMockClock(t)
mockMail := mocks.NewMockMailService(t)
underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, &types.ServerSettings{}) underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, mockMail, &types.ServerSettings{})
weakPasswords := []string{ weakPasswords := []string{
"123!ab", // too short "123!ab", // too short
@@ -148,6 +156,7 @@ func TestSignUp(t *testing.T) {
mockDbAuth := mocks.NewMockDbAuth(t) mockDbAuth := mocks.NewMockDbAuth(t)
mockRandom := mocks.NewMockRandomGenerator(t) mockRandom := mocks.NewMockRandomGenerator(t)
mockClock := mocks.NewMockClock(t) mockClock := mocks.NewMockClock(t)
mockMail := mocks.NewMockMailService(t)
expected := User{ expected := User{
Id: uuid.New(), Id: uuid.New(),
@@ -169,7 +178,7 @@ func TestSignUp(t *testing.T) {
mockDbAuth.EXPECT().InsertUser(db.NewUser(expected.Id, expected.Email, false, nil, false, GetHashPassword(password, salt), salt, createTime)).Return(nil) mockDbAuth.EXPECT().InsertUser(db.NewUser(expected.Id, expected.Email, false, nil, false, GetHashPassword(password, salt), salt, createTime)).Return(nil)
underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, &types.ServerSettings{}) underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, mockMail, &types.ServerSettings{})
actual, err := underTest.SignUp(expected.Email, password) actual, err := underTest.SignUp(expected.Email, password)
@@ -177,4 +186,61 @@ func TestSignUp(t *testing.T) {
assert.Equal(t, expected, *actual) assert.Equal(t, expected, *actual)
}) })
t.Run("should return ErrAccountExists", func(t *testing.T) {
t.Parallel()
mockDbAuth := mocks.NewMockDbAuth(t)
mockRandom := mocks.NewMockRandomGenerator(t)
mockClock := mocks.NewMockClock(t)
mockMail := mocks.NewMockMailService(t)
user := User{
Id: uuid.New(),
Email: "some@valid.email",
}
random := NewRandomGeneratorImpl()
salt, err := random.Bytes(16)
assert.Nil(t, err)
password := "SomeStrongPassword123!"
mockRandom.EXPECT().UUID().Return(user.Id, nil)
mockRandom.EXPECT().Bytes(16).Return(salt, nil)
createTime := time.Date(2020, 1, 1, 0, 0, 0, 0, time.UTC)
mockClock.EXPECT().Now().Return(createTime)
mockDbAuth.EXPECT().InsertUser(db.NewUser(user.Id, user.Email, false, nil, false, GetHashPassword(password, salt), salt, createTime)).Return(db.ErrUserExists)
underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, mockMail, &types.ServerSettings{})
_, err = underTest.SignUp(user.Email, password)
assert.Equal(t, ErrAccountExists, err)
})
}
func TestSendVerificationMail(t *testing.T) {
t.Parallel()
t.Run("should use stored token and send mail", func(t *testing.T) {
t.Parallel()
token := "someRandomTokenToUse"
email := "some@email.de"
userId := uuid.New()
mockDbAuth := mocks.NewMockDbAuth(t)
mockRandom := mocks.NewMockRandomGenerator(t)
mockClock := mocks.NewMockClock(t)
mockMail := mocks.NewMockMailService(t)
mockDbAuth.EXPECT().GetEmailVerificationToken(userId).Return(token, nil)
mockMail.EXPECT().SendMail(email, "Welcome to ME-FIT", mock.MatchedBy(func(message string) bool { return strings.Contains(message, token) })).Return(nil)
underTest := NewServiceAuthImpl(mockDbAuth, mockRandom, mockClock, mockMail, &types.ServerSettings{})
underTest.SendVerificationMail(userId, email)
})
} }

View File

@@ -6,15 +6,19 @@ import (
"net/smtp" "net/smtp"
) )
type MailService struct { type MailService interface {
SendMail(to string, subject string, message string) error
}
type MailServiceImpl struct {
serverSettings *types.ServerSettings serverSettings *types.ServerSettings
} }
func NewMailService(serverSettings *types.ServerSettings) MailService { func NewMailServiceImpl(serverSettings *types.ServerSettings) MailServiceImpl {
return MailService{serverSettings: serverSettings} return MailServiceImpl{serverSettings: serverSettings}
} }
func (m MailService) SendMail(to string, subject string, message string) error { func (m MailServiceImpl) SendMail(to string, subject string, message string) error {
if m.serverSettings.Smtp == nil { if m.serverSettings.Smtp == nil {
return nil return nil
} }