mirror of
https://github.com/muety/wakapi.git
synced 2023-08-10 21:12:56 +03:00
468 lines
17 KiB
Go
468 lines
17 KiB
Go
package services
|
|
|
|
import (
|
|
"github.com/muety/wakapi/mocks"
|
|
"github.com/muety/wakapi/models"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/mock"
|
|
"github.com/stretchr/testify/suite"
|
|
"math/rand"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
const (
|
|
TestUserId = "muety"
|
|
TestProject1 = "test-project-1"
|
|
TestProject2 = "test-project-2"
|
|
TestProjectLabel1 = "private"
|
|
TestProjectLabel2 = "work"
|
|
TestProjectLabel3 = "non-existing"
|
|
TestLanguageGo = "Go"
|
|
TestLanguageJava = "Java"
|
|
TestLanguagePython = "Python"
|
|
TestEditorGoland = "GoLand"
|
|
TestEditorIntellij = "idea"
|
|
TestEditorVscode = "vscode"
|
|
TestOsLinux = "Linux"
|
|
TestOsWin = "Windows"
|
|
TestMachine1 = "muety-desktop"
|
|
TestMachine2 = "muety-work"
|
|
MinUnixTime1 = 1601510400000 * 1e6
|
|
)
|
|
|
|
type SummaryServiceTestSuite struct {
|
|
suite.Suite
|
|
TestUser *models.User
|
|
TestStartTime time.Time
|
|
TestHeartbeats []*models.Heartbeat
|
|
TestLabels []*models.ProjectLabel
|
|
SummaryRepository *mocks.SummaryRepositoryMock
|
|
HeartbeatService *mocks.HeartbeatServiceMock
|
|
AliasService *mocks.AliasServiceMock
|
|
ProjectLabelService *mocks.ProjectLabelServiceMock
|
|
}
|
|
|
|
func (suite *SummaryServiceTestSuite) SetupSuite() {
|
|
suite.TestUser = &models.User{ID: TestUserId}
|
|
|
|
suite.TestStartTime = time.Unix(0, MinUnixTime1)
|
|
suite.TestHeartbeats = []*models.Heartbeat{
|
|
{
|
|
ID: rand.Uint64(),
|
|
UserID: TestUserId,
|
|
Project: TestProject1,
|
|
Language: TestLanguageGo,
|
|
Editor: TestEditorGoland,
|
|
OperatingSystem: TestOsLinux,
|
|
Machine: TestMachine1,
|
|
Time: models.CustomTime(suite.TestStartTime),
|
|
},
|
|
{
|
|
ID: rand.Uint64(),
|
|
UserID: TestUserId,
|
|
Project: TestProject1,
|
|
Language: TestLanguageGo,
|
|
Editor: TestEditorGoland,
|
|
OperatingSystem: TestOsLinux,
|
|
Machine: TestMachine1,
|
|
Time: models.CustomTime(suite.TestStartTime.Add(30 * time.Second)),
|
|
},
|
|
{
|
|
ID: rand.Uint64(),
|
|
UserID: TestUserId,
|
|
Project: TestProject1,
|
|
Language: TestLanguageGo,
|
|
Editor: TestEditorVscode,
|
|
OperatingSystem: TestOsLinux,
|
|
Machine: TestMachine1,
|
|
Time: models.CustomTime(suite.TestStartTime.Add(3 * time.Minute)),
|
|
},
|
|
}
|
|
suite.TestLabels = []*models.ProjectLabel{
|
|
{
|
|
ID: uint(rand.Uint32()),
|
|
UserID: TestUserId,
|
|
ProjectKey: TestProject1,
|
|
Label: TestProjectLabel1,
|
|
},
|
|
{
|
|
ID: uint(rand.Uint32()),
|
|
UserID: TestUserId,
|
|
ProjectKey: TestProjectLabel3,
|
|
Label: "blaahh",
|
|
},
|
|
}
|
|
}
|
|
|
|
func (suite *SummaryServiceTestSuite) BeforeTest(suiteName, testName string) {
|
|
suite.SummaryRepository = new(mocks.SummaryRepositoryMock)
|
|
suite.HeartbeatService = new(mocks.HeartbeatServiceMock)
|
|
suite.AliasService = new(mocks.AliasServiceMock)
|
|
suite.ProjectLabelService = new(mocks.ProjectLabelServiceMock)
|
|
}
|
|
|
|
func TestSummaryServiceTestSuite(t *testing.T) {
|
|
suite.Run(t, new(SummaryServiceTestSuite))
|
|
}
|
|
|
|
func (suite *SummaryServiceTestSuite) TestSummaryService_Summarize() {
|
|
sut := NewSummaryService(suite.SummaryRepository, suite.HeartbeatService, suite.AliasService, suite.ProjectLabelService)
|
|
|
|
var (
|
|
from time.Time
|
|
to time.Time
|
|
result *models.Summary
|
|
err error
|
|
)
|
|
|
|
/* TEST 1 */
|
|
from, to = suite.TestStartTime.Add(-1*time.Hour), suite.TestStartTime.Add(-1*time.Minute)
|
|
suite.HeartbeatService.On("GetAllWithin", from, to, suite.TestUser).Return(filter(from, to, suite.TestHeartbeats), nil)
|
|
|
|
result, err = sut.Summarize(from, to, suite.TestUser)
|
|
|
|
assert.Nil(suite.T(), err)
|
|
assert.NotNil(suite.T(), result)
|
|
assert.Equal(suite.T(), from, result.FromTime.T())
|
|
assert.Equal(suite.T(), to, result.ToTime.T())
|
|
assert.Zero(suite.T(), result.TotalTime())
|
|
assert.Empty(suite.T(), result.Projects)
|
|
|
|
/* TEST 2 */
|
|
from, to = suite.TestStartTime.Add(-1*time.Hour), suite.TestStartTime.Add(1*time.Second)
|
|
suite.HeartbeatService.On("GetAllWithin", from, to, suite.TestUser).Return(filter(from, to, suite.TestHeartbeats), nil)
|
|
|
|
result, err = sut.Summarize(from, to, suite.TestUser)
|
|
|
|
assert.Nil(suite.T(), err)
|
|
assert.NotNil(suite.T(), result)
|
|
assert.Equal(suite.T(), suite.TestHeartbeats[0].Time.T(), result.FromTime.T())
|
|
assert.Equal(suite.T(), suite.TestHeartbeats[0].Time.T(), result.ToTime.T())
|
|
assert.Zero(suite.T(), result.TotalTime())
|
|
assertNumAllItems(suite.T(), 1, result, "")
|
|
|
|
/* TEST 3 */
|
|
from, to = suite.TestStartTime, suite.TestStartTime.Add(1*time.Hour)
|
|
suite.HeartbeatService.On("GetAllWithin", from, to, suite.TestUser).Return(filter(from, to, suite.TestHeartbeats), nil)
|
|
|
|
result, err = sut.Summarize(from, to, suite.TestUser)
|
|
|
|
assert.Nil(suite.T(), err)
|
|
assert.NotNil(suite.T(), result)
|
|
assert.Equal(suite.T(), suite.TestHeartbeats[0].Time.T(), result.FromTime.T())
|
|
assert.Equal(suite.T(), suite.TestHeartbeats[len(suite.TestHeartbeats)-1].Time.T(), result.ToTime.T())
|
|
assert.Equal(suite.T(), 150*time.Second, result.TotalTime())
|
|
assert.Equal(suite.T(), 30*time.Second, result.TotalTimeByKey(models.SummaryEditor, TestEditorGoland))
|
|
assert.Equal(suite.T(), 120*time.Second, result.TotalTimeByKey(models.SummaryEditor, TestEditorVscode))
|
|
assert.Len(suite.T(), result.Editors, 2)
|
|
assertNumAllItems(suite.T(), 1, result, "e")
|
|
}
|
|
|
|
func (suite *SummaryServiceTestSuite) TestSummaryService_Retrieve() {
|
|
sut := NewSummaryService(suite.SummaryRepository, suite.HeartbeatService, suite.AliasService, suite.ProjectLabelService)
|
|
|
|
var (
|
|
summaries []*models.Summary
|
|
from time.Time
|
|
to time.Time
|
|
result *models.Summary
|
|
err error
|
|
)
|
|
|
|
/* TEST 1 */
|
|
from, to = suite.TestStartTime.Add(-12*time.Hour), suite.TestStartTime.Add(12*time.Hour)
|
|
summaries = []*models.Summary{
|
|
{
|
|
ID: uint(rand.Uint32()),
|
|
UserID: TestUserId,
|
|
FromTime: models.CustomTime(from.Add(10 * time.Minute)),
|
|
ToTime: models.CustomTime(to.Add(-10 * time.Minute)),
|
|
Projects: []*models.SummaryItem{
|
|
{
|
|
Type: models.SummaryProject,
|
|
Key: TestProject1,
|
|
Total: 45 * time.Minute / time.Second, // hack
|
|
},
|
|
},
|
|
Languages: []*models.SummaryItem{},
|
|
Editors: []*models.SummaryItem{},
|
|
OperatingSystems: []*models.SummaryItem{},
|
|
Machines: []*models.SummaryItem{},
|
|
},
|
|
}
|
|
|
|
suite.SummaryRepository.On("GetByUserWithin", suite.TestUser, from, to).Return(summaries, nil)
|
|
suite.HeartbeatService.On("GetAllWithin", from, summaries[0].FromTime.T(), suite.TestUser).Return([]*models.Heartbeat{}, nil)
|
|
suite.HeartbeatService.On("GetAllWithin", summaries[0].ToTime.T(), to, suite.TestUser).Return([]*models.Heartbeat{}, nil)
|
|
|
|
result, err = sut.Retrieve(from, to, suite.TestUser)
|
|
|
|
assert.Nil(suite.T(), err)
|
|
assert.NotNil(suite.T(), result)
|
|
assert.Len(suite.T(), result.Projects, 1)
|
|
assert.Equal(suite.T(), summaries[0].Projects[0].Total*time.Second, result.TotalTime())
|
|
suite.HeartbeatService.AssertNumberOfCalls(suite.T(), "GetAllWithin", 2)
|
|
|
|
/* TEST 2 */
|
|
from, to = suite.TestStartTime.Add(-10*time.Minute), suite.TestStartTime.Add(12*time.Hour)
|
|
summaries = []*models.Summary{
|
|
{
|
|
ID: uint(rand.Uint32()),
|
|
UserID: TestUserId,
|
|
FromTime: models.CustomTime(from.Add(20 * time.Minute)),
|
|
ToTime: models.CustomTime(to.Add(-6 * time.Hour)),
|
|
Projects: []*models.SummaryItem{
|
|
{
|
|
Type: models.SummaryProject,
|
|
Key: TestProject1,
|
|
Total: 45 * time.Minute / time.Second, // hack
|
|
},
|
|
},
|
|
Languages: []*models.SummaryItem{},
|
|
Editors: []*models.SummaryItem{},
|
|
OperatingSystems: []*models.SummaryItem{},
|
|
Machines: []*models.SummaryItem{},
|
|
},
|
|
{
|
|
ID: uint(rand.Uint32()),
|
|
UserID: TestUserId,
|
|
FromTime: models.CustomTime(to.Add(-6 * time.Hour)),
|
|
ToTime: models.CustomTime(to),
|
|
Projects: []*models.SummaryItem{
|
|
{
|
|
Type: models.SummaryProject,
|
|
Key: TestProject2,
|
|
Total: 45 * time.Minute / time.Second, // hack
|
|
},
|
|
},
|
|
Languages: []*models.SummaryItem{},
|
|
Editors: []*models.SummaryItem{},
|
|
OperatingSystems: []*models.SummaryItem{},
|
|
Machines: []*models.SummaryItem{},
|
|
},
|
|
}
|
|
|
|
suite.SummaryRepository.On("GetByUserWithin", suite.TestUser, from, to).Return(summaries, nil)
|
|
suite.HeartbeatService.On("GetAllWithin", from, summaries[0].FromTime.T(), suite.TestUser).Return(filter(from, summaries[0].FromTime.T(), suite.TestHeartbeats), nil)
|
|
|
|
result, err = sut.Retrieve(from, to, suite.TestUser)
|
|
|
|
assert.Nil(suite.T(), err)
|
|
assert.NotNil(suite.T(), result)
|
|
assert.Len(suite.T(), result.Projects, 2)
|
|
assert.Equal(suite.T(), 150*time.Second+90*time.Minute, result.TotalTime())
|
|
assert.Equal(suite.T(), 150*time.Second+45*time.Minute, result.TotalTimeByKey(models.SummaryProject, TestProject1))
|
|
assert.Equal(suite.T(), 45*time.Minute, result.TotalTimeByKey(models.SummaryProject, TestProject2))
|
|
suite.HeartbeatService.AssertNumberOfCalls(suite.T(), "GetAllWithin", 2+1)
|
|
|
|
/* TEST 3 */
|
|
from = time.Date(suite.TestStartTime.Year(), suite.TestStartTime.Month(), suite.TestStartTime.Day()+1, 0, 0, 0, 0, suite.TestStartTime.Location()) // start of next day
|
|
to = time.Date(from.Year(), from.Month(), from.Day()+2, 13, 30, 0, 0, from.Location()) // noon of third-next day
|
|
summaries = []*models.Summary{
|
|
{
|
|
ID: uint(rand.Uint32()),
|
|
UserID: TestUserId,
|
|
FromTime: models.CustomTime(from),
|
|
ToTime: models.CustomTime(from.Add(24 * time.Hour)),
|
|
Projects: []*models.SummaryItem{
|
|
{
|
|
Type: models.SummaryProject,
|
|
Key: TestProject1,
|
|
Total: 45 * time.Minute / time.Second, // hack
|
|
},
|
|
},
|
|
Languages: []*models.SummaryItem{},
|
|
Editors: []*models.SummaryItem{},
|
|
OperatingSystems: []*models.SummaryItem{},
|
|
Machines: []*models.SummaryItem{},
|
|
},
|
|
{
|
|
ID: uint(rand.Uint32()),
|
|
UserID: TestUserId,
|
|
FromTime: models.CustomTime(to.Add(-2 * time.Hour)),
|
|
ToTime: models.CustomTime(to),
|
|
Projects: []*models.SummaryItem{
|
|
{
|
|
Type: models.SummaryProject,
|
|
Key: TestProject2,
|
|
Total: 45 * time.Minute / time.Second, // hack
|
|
},
|
|
},
|
|
Languages: []*models.SummaryItem{},
|
|
Editors: []*models.SummaryItem{},
|
|
OperatingSystems: []*models.SummaryItem{},
|
|
Machines: []*models.SummaryItem{},
|
|
},
|
|
}
|
|
|
|
suite.SummaryRepository.On("GetByUserWithin", suite.TestUser, from, to).Return(summaries, nil)
|
|
suite.HeartbeatService.On("GetAllWithin", summaries[0].ToTime.T(), summaries[1].FromTime.T(), suite.TestUser).Return(filter(summaries[0].ToTime.T(), summaries[1].FromTime.T(), suite.TestHeartbeats), nil)
|
|
|
|
result, err = sut.Retrieve(from, to, suite.TestUser)
|
|
|
|
assert.Nil(suite.T(), err)
|
|
assert.NotNil(suite.T(), result)
|
|
assert.Len(suite.T(), result.Projects, 2)
|
|
assert.Equal(suite.T(), 90*time.Minute, result.TotalTime())
|
|
assert.Equal(suite.T(), 45*time.Minute, result.TotalTimeByKey(models.SummaryProject, TestProject1))
|
|
assert.Equal(suite.T(), 45*time.Minute, result.TotalTimeByKey(models.SummaryProject, TestProject2))
|
|
suite.HeartbeatService.AssertNumberOfCalls(suite.T(), "GetAllWithin", 2+1+1)
|
|
}
|
|
|
|
func (suite *SummaryServiceTestSuite) TestSummaryService_Retrieve_DuplicateSummaries() {
|
|
sut := NewSummaryService(suite.SummaryRepository, suite.HeartbeatService, suite.AliasService, suite.ProjectLabelService)
|
|
|
|
suite.ProjectLabelService.On("GetByUser", suite.TestUser.ID).Return([]*models.ProjectLabel{}, nil)
|
|
|
|
var (
|
|
summaries []*models.Summary
|
|
from time.Time
|
|
to time.Time
|
|
result *models.Summary
|
|
err error
|
|
)
|
|
|
|
from, to = suite.TestStartTime.Add(-12*time.Hour), suite.TestStartTime.Add(12*time.Hour)
|
|
summaries = []*models.Summary{
|
|
{
|
|
ID: uint(rand.Uint32()),
|
|
UserID: TestUserId,
|
|
FromTime: models.CustomTime(from.Add(10 * time.Minute)),
|
|
ToTime: models.CustomTime(to.Add(-10 * time.Minute)),
|
|
Projects: []*models.SummaryItem{
|
|
{
|
|
Type: models.SummaryProject,
|
|
Key: TestProject1,
|
|
Total: 45 * time.Minute / time.Second, // hack
|
|
},
|
|
},
|
|
Languages: []*models.SummaryItem{},
|
|
Editors: []*models.SummaryItem{},
|
|
OperatingSystems: []*models.SummaryItem{},
|
|
Machines: []*models.SummaryItem{},
|
|
},
|
|
}
|
|
summaries = append(summaries, &(*summaries[0])) // add same summary again -> mustn't be counted twice!
|
|
|
|
suite.SummaryRepository.On("GetByUserWithin", suite.TestUser, from, to).Return(summaries, nil)
|
|
suite.HeartbeatService.On("GetAllWithin", from, summaries[0].FromTime.T(), suite.TestUser).Return([]*models.Heartbeat{}, nil)
|
|
suite.HeartbeatService.On("GetAllWithin", summaries[0].ToTime.T(), to, suite.TestUser).Return([]*models.Heartbeat{}, nil)
|
|
|
|
result, err = sut.Retrieve(from, to, suite.TestUser)
|
|
|
|
assert.Nil(suite.T(), err)
|
|
assert.NotNil(suite.T(), result)
|
|
assert.Len(suite.T(), result.Projects, 1)
|
|
assert.Equal(suite.T(), summaries[0].Projects[0].Total*time.Second, result.TotalTime())
|
|
suite.HeartbeatService.AssertNumberOfCalls(suite.T(), "GetAllWithin", 2)
|
|
}
|
|
|
|
func (suite *SummaryServiceTestSuite) TestSummaryService_Aliased() {
|
|
sut := NewSummaryService(suite.SummaryRepository, suite.HeartbeatService, suite.AliasService, suite.ProjectLabelService)
|
|
|
|
suite.ProjectLabelService.On("GetByUser", suite.TestUser.ID).Return([]*models.ProjectLabel{}, nil)
|
|
|
|
var (
|
|
from time.Time
|
|
to time.Time
|
|
result *models.Summary
|
|
err error
|
|
)
|
|
|
|
from, to = suite.TestStartTime, suite.TestStartTime.Add(1*time.Hour)
|
|
|
|
heartbeats := filter(from, to, suite.TestHeartbeats)
|
|
heartbeats = append(heartbeats, &models.Heartbeat{
|
|
ID: rand.Uint64(),
|
|
UserID: TestUserId,
|
|
Project: TestProject2,
|
|
Language: TestLanguageGo,
|
|
Editor: TestEditorGoland,
|
|
OperatingSystem: TestOsLinux,
|
|
Machine: TestMachine1,
|
|
Time: models.CustomTime(heartbeats[len(heartbeats)-1].Time.T().Add(10 * time.Second)),
|
|
})
|
|
|
|
suite.HeartbeatService.On("GetAllWithin", from, to, suite.TestUser).Return(heartbeats, nil)
|
|
suite.AliasService.On("InitializeUser", TestUserId).Return(nil)
|
|
suite.AliasService.On("GetAliasOrDefault", TestUserId, mock.Anything, TestProject1).Return(TestProject2, nil)
|
|
suite.AliasService.On("GetAliasOrDefault", TestUserId, mock.Anything, TestProject2).Return(TestProject2, nil)
|
|
suite.AliasService.On("GetAliasOrDefault", TestUserId, mock.Anything, mock.Anything).Return("", nil)
|
|
suite.ProjectLabelService.On("GetByUser", suite.TestUser.ID).Return(suite.TestLabels, nil).Once()
|
|
|
|
result, err = sut.Aliased(from, to, suite.TestUser, sut.Summarize, false)
|
|
|
|
assert.Nil(suite.T(), err)
|
|
assert.NotNil(suite.T(), result)
|
|
assert.Zero(suite.T(), result.TotalTimeByKey(models.SummaryProject, TestProject1))
|
|
assert.NotZero(suite.T(), result.TotalTimeByKey(models.SummaryProject, TestProject2))
|
|
}
|
|
|
|
func (suite *SummaryServiceTestSuite) TestSummaryService_Aliased_ProjectLabels() {
|
|
sut := NewSummaryService(suite.SummaryRepository, suite.HeartbeatService, suite.AliasService, suite.ProjectLabelService)
|
|
|
|
var (
|
|
from time.Time
|
|
to time.Time
|
|
result *models.Summary
|
|
err error
|
|
)
|
|
|
|
from, to = suite.TestStartTime, suite.TestStartTime.Add(1*time.Hour)
|
|
|
|
heartbeats := filter(from, to, suite.TestHeartbeats)
|
|
heartbeats = append(heartbeats, &models.Heartbeat{
|
|
ID: rand.Uint64(),
|
|
UserID: TestUserId,
|
|
Project: TestProject2,
|
|
Language: TestLanguageGo,
|
|
Editor: TestEditorGoland,
|
|
OperatingSystem: TestOsLinux,
|
|
Machine: TestMachine1,
|
|
Time: models.CustomTime(heartbeats[len(heartbeats)-1].Time.T().Add(10 * time.Second)),
|
|
})
|
|
|
|
suite.ProjectLabelService.On("GetByUser", suite.TestUser.ID).Return(suite.TestLabels, nil).Once()
|
|
suite.HeartbeatService.On("GetAllWithin", from, to, suite.TestUser).Return(heartbeats, nil)
|
|
suite.AliasService.On("InitializeUser", TestUserId).Return(nil)
|
|
suite.AliasService.On("GetAliasOrDefault", TestUserId, mock.Anything, TestProject1).Return(TestProject1, nil)
|
|
suite.AliasService.On("GetAliasOrDefault", TestUserId, mock.Anything, TestProject2).Return(TestProject1, nil)
|
|
suite.AliasService.On("GetAliasOrDefault", TestUserId, mock.Anything, mock.Anything).Return("", nil)
|
|
|
|
result, err = sut.Aliased(from, to, suite.TestUser, sut.Summarize, false)
|
|
|
|
assert.Nil(suite.T(), err)
|
|
assert.NotNil(suite.T(), result)
|
|
assert.Equal(suite.T(), 160*time.Second, result.TotalTimeByKey(models.SummaryLabel, TestProjectLabel1))
|
|
}
|
|
|
|
func filter(from, to time.Time, heartbeats []*models.Heartbeat) []*models.Heartbeat {
|
|
filtered := make([]*models.Heartbeat, 0, len(heartbeats))
|
|
for _, h := range heartbeats {
|
|
if (h.Time.T().Equal(from) || h.Time.T().After(from)) && h.Time.T().Before(to) {
|
|
filtered = append(filtered, h)
|
|
}
|
|
}
|
|
return filtered
|
|
}
|
|
|
|
func assertNumAllItems(t *testing.T, expected int, summary *models.Summary, except string) {
|
|
if !strings.Contains(except, "p") {
|
|
assert.Len(t, summary.Projects, expected)
|
|
}
|
|
if !strings.Contains(except, "e") {
|
|
assert.Len(t, summary.Editors, expected)
|
|
}
|
|
if !strings.Contains(except, "l") {
|
|
assert.Len(t, summary.Languages, expected)
|
|
}
|
|
if !strings.Contains(except, "o") {
|
|
assert.Len(t, summary.OperatingSystems, expected)
|
|
}
|
|
if !strings.Contains(except, "m") {
|
|
assert.Len(t, summary.Machines, expected)
|
|
}
|
|
}
|