123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315 |
- package handler
- import (
- "context"
- "database/sql"
- "net/http/httptest"
- "testing"
- "github.com/brianvoe/gofakeit/v6"
- "github.com/gofiber/fiber/v2"
- "github.com/gojuno/minimock/v3"
- "github.com/stretchr/testify/assert"
- "git.dmitriygnatenko.ru/dima/dmitriygnatenko-v2/internal/helper/test"
- "git.dmitriygnatenko.ru/dima/dmitriygnatenko-v2/internal/model"
- "git.dmitriygnatenko.ru/dima/dmitriygnatenko-v2/internal/service/handler/mocks"
- )
- func TestArticleHandler(t *testing.T) {
- t.Parallel()
- type req struct {
- method string
- route string
- }
- var (
- articleID = gofakeit.Uint64()
- articleURL = gofakeit.Word()
- publishTime = gofakeit.Date()
- internalErr = gofakeit.Error()
- language = model.LanguageID(gofakeit.Uint64())
- article = model.Article{
- ID: articleID,
- URL: articleURL,
- Title: gofakeit.Phrase(),
- Text: gofakeit.Phrase(),
- PublishTime: publishTime,
- PreviewText: sql.NullString{Valid: true, String: gofakeit.Phrase()},
- Image: gofakeit.URL(),
- Language: language,
- IsActive: true,
- }
- notActiveArticle = model.Article{
- ID: articleID,
- URL: articleURL,
- Title: gofakeit.Phrase(),
- Text: gofakeit.Phrase(),
- PublishTime: publishTime,
- PreviewText: sql.NullString{Valid: true, String: gofakeit.Phrase()},
- Image: gofakeit.URL(),
- Language: language,
- }
- tags = []model.Tag{
- {
- ID: gofakeit.Uint64(),
- Tag: gofakeit.Word(),
- URL: gofakeit.Word(),
- },
- {
- ID: gofakeit.Uint64(),
- Tag: gofakeit.Word(),
- URL: gofakeit.Word(),
- },
- }
- previewArticles = []model.ArticlePreview{
- {
- ID: gofakeit.Uint64(),
- URL: gofakeit.URL(),
- Title: gofakeit.Phrase(),
- PublishTime: publishTime,
- PreviewText: sql.NullString{Valid: true, String: gofakeit.Phrase()},
- Image: gofakeit.URL(),
- },
- {
- ID: gofakeit.Uint64(),
- URL: gofakeit.URL(),
- Title: gofakeit.Phrase(),
- PublishTime: publishTime,
- PreviewText: sql.NullString{Valid: true, String: gofakeit.Phrase()},
- Image: gofakeit.URL(),
- },
- {
- ID: gofakeit.Uint64(),
- URL: gofakeit.URL(),
- Title: gofakeit.Phrase(),
- PublishTime: publishTime,
- PreviewText: sql.NullString{Valid: true, String: gofakeit.Phrase()},
- Image: gofakeit.URL(),
- },
- {
- ID: gofakeit.Uint64(),
- URL: gofakeit.URL(),
- Title: gofakeit.Phrase(),
- PublishTime: publishTime,
- PreviewText: sql.NullString{Valid: true, String: gofakeit.Phrase()},
- Image: gofakeit.URL(),
- },
- }
- )
- tests := []struct {
- name string
- req req
- res int
- err error
- cacheMock func(mc *minimock.Controller) CacheService
- tagMock func(mc *minimock.Controller) TagRepository
- articleMock func(mc *minimock.Controller) ArticleRepository
- }{
- {
- name: "positive case",
- req: req{
- method: fiber.MethodGet,
- route: "/article/" + articleURL,
- },
- res: fiber.StatusOK,
- err: nil,
- cacheMock: func(mc *minimock.Controller) CacheService {
- mock := mocks.NewCacheServiceMock(mc)
- mock.GetMock.When(ArticleCacheKey+articleURL).Then(nil, false)
- mock.GetMock.When(RecentArticlesCacheKey).Then(nil, false)
- mock.GetMock.When(UsedTagsCacheKey).Then(nil, false)
- mock.SetMock.Return()
- return mock
- },
- tagMock: func(mc *minimock.Controller) TagRepository {
- mock := mocks.NewTagRepositoryMock(mc)
- mock.GetAllUsedMock.Return(tags, nil)
- return mock
- },
- articleMock: func(mc *minimock.Controller) ArticleRepository {
- mock := mocks.NewArticleRepositoryMock(mc)
- mock.GetByURLMock.Inspect(func(ctx context.Context, url string, lang model.LanguageID) {
- assert.Equal(mc, articleURL, url)
- }).Return(&article, nil)
- mock.GetAllPreviewMock.Return(previewArticles, nil)
- return mock
- },
- },
- {
- name: "negative case - article not found",
- req: req{
- method: fiber.MethodGet,
- route: "/article/" + articleURL,
- },
- res: fiber.StatusNotFound,
- err: nil,
- cacheMock: func(mc *minimock.Controller) CacheService {
- mock := mocks.NewCacheServiceMock(mc)
- mock.GetMock.Expect(ArticleCacheKey+articleURL).Return(nil, false)
- return mock
- },
- tagMock: func(mc *minimock.Controller) TagRepository {
- return mocks.NewTagRepositoryMock(mc)
- },
- articleMock: func(mc *minimock.Controller) ArticleRepository {
- mock := mocks.NewArticleRepositoryMock(mc)
- mock.GetByURLMock.Inspect(func(ctx context.Context, url string, lang model.LanguageID) {
- assert.Equal(mc, articleURL, url)
- }).Return(nil, sql.ErrNoRows)
- return mock
- },
- },
- {
- name: "negative case - article repository error",
- req: req{
- method: fiber.MethodGet,
- route: "/article/" + articleURL,
- },
- res: fiber.StatusInternalServerError,
- err: nil,
- cacheMock: func(mc *minimock.Controller) CacheService {
- mock := mocks.NewCacheServiceMock(mc)
- mock.GetMock.Expect(ArticleCacheKey+articleURL).Return(nil, false)
- return mock
- },
- tagMock: func(mc *minimock.Controller) TagRepository {
- return mocks.NewTagRepositoryMock(mc)
- },
- articleMock: func(mc *minimock.Controller) ArticleRepository {
- mock := mocks.NewArticleRepositoryMock(mc)
- mock.GetByURLMock.Inspect(func(ctx context.Context, url string, lang model.LanguageID) {
- assert.Equal(mc, articleURL, url)
- }).Return(nil, internalErr)
- return mock
- },
- },
- {
- name: "negative case - article not active",
- req: req{
- method: fiber.MethodGet,
- route: "/article/" + articleURL,
- },
- res: fiber.StatusNotFound,
- err: nil,
- cacheMock: func(mc *minimock.Controller) CacheService {
- mock := mocks.NewCacheServiceMock(mc)
- mock.GetMock.Expect(ArticleCacheKey+articleURL).Return(nil, false)
- return mock
- },
- tagMock: func(mc *minimock.Controller) TagRepository {
- return mocks.NewTagRepositoryMock(mc)
- },
- articleMock: func(mc *minimock.Controller) ArticleRepository {
- mock := mocks.NewArticleRepositoryMock(mc)
- mock.GetByURLMock.Inspect(func(ctx context.Context, url string, lang model.LanguageID) {
- assert.Equal(mc, articleURL, url)
- }).Return(¬ActiveArticle, nil)
- return mock
- },
- },
- {
- name: "negative case - articles repository error",
- req: req{
- method: fiber.MethodGet,
- route: "/article/" + articleURL,
- },
- res: fiber.StatusInternalServerError,
- err: nil,
- cacheMock: func(mc *minimock.Controller) CacheService {
- mock := mocks.NewCacheServiceMock(mc)
- mock.GetMock.When(ArticleCacheKey+articleURL).Then(nil, false)
- mock.GetMock.When(RecentArticlesCacheKey).Then(nil, false)
- mock.SetMock.Return()
- return mock
- },
- tagMock: func(mc *minimock.Controller) TagRepository {
- return mocks.NewTagRepositoryMock(mc)
- },
- articleMock: func(mc *minimock.Controller) ArticleRepository {
- mock := mocks.NewArticleRepositoryMock(mc)
- mock.GetByURLMock.Inspect(func(ctx context.Context, url string, lang model.LanguageID) {
- assert.Equal(mc, articleURL, url)
- }).Return(&article, nil)
- mock.GetAllPreviewMock.Return(nil, internalErr)
- return mock
- },
- },
- {
- name: "negative case - tags repository error",
- req: req{
- method: fiber.MethodGet,
- route: "/article/" + articleURL,
- },
- res: fiber.StatusInternalServerError,
- err: nil,
- cacheMock: func(mc *minimock.Controller) CacheService {
- mock := mocks.NewCacheServiceMock(mc)
- mock.GetMock.When(ArticleCacheKey+articleURL).Then(nil, false)
- mock.GetMock.When(RecentArticlesCacheKey).Then(nil, false)
- mock.GetMock.When(UsedTagsCacheKey).Then(nil, false)
- mock.SetMock.Return()
- return mock
- },
- tagMock: func(mc *minimock.Controller) TagRepository {
- mock := mocks.NewTagRepositoryMock(mc)
- mock.GetAllUsedMock.Return(nil, internalErr)
- return mock
- },
- articleMock: func(mc *minimock.Controller) ArticleRepository {
- mock := mocks.NewArticleRepositoryMock(mc)
- mock.GetByURLMock.Inspect(func(ctx context.Context, url string, lang model.LanguageID) {
- assert.Equal(mc, articleURL, url)
- }).Return(&article, nil)
- mock.GetAllPreviewMock.Return(previewArticles, nil)
- return mock
- },
- },
- }
- for _, tt := range tests {
- t.Run(tt.name, func(t *testing.T) {
- t.Parallel()
- mc := minimock.NewController(t)
- fiberApp := fiber.New(test.GetFiberTestConfig())
- fiberReq := httptest.NewRequest(tt.req.method, tt.req.route, nil)
- fiberApp.Get("/article/:article", NewArticlePageHandler(
- tt.cacheMock(mc),
- tt.articleMock(mc),
- tt.tagMock(mc),
- ))
- fiberRes, fiberErr := fiberApp.Test(fiberReq)
- assert.Equal(t, tt.res, fiberRes.StatusCode)
- assert.Equal(t, tt.err, fiberErr)
- })
- }
- }
|