2020-11-14 14:30:45 +03:00
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"
2021-06-12 12:26:15 +03:00
TestProjectLabel1 = "private"
TestProjectLabel2 = "work"
TestProjectLabel3 = "non-existing"
2020-11-14 14:30:45 +03:00
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
2021-06-12 12:09:24 +03:00
TestUser * models . User
TestStartTime time . Time
TestHeartbeats [ ] * models . Heartbeat
2021-06-12 12:26:15 +03:00
TestLabels [ ] * models . ProjectLabel
2021-06-12 12:09:24 +03:00
SummaryRepository * mocks . SummaryRepositoryMock
HeartbeatService * mocks . HeartbeatServiceMock
AliasService * mocks . AliasServiceMock
ProjectLabelService * mocks . ProjectLabelServiceMock
2020-11-14 14:30:45 +03:00
}
func ( suite * SummaryServiceTestSuite ) SetupSuite ( ) {
suite . TestUser = & models . User { ID : TestUserId }
suite . TestStartTime = time . Unix ( 0 , MinUnixTime1 )
suite . TestHeartbeats = [ ] * models . Heartbeat {
{
2021-12-15 14:52:24 +03:00
ID : rand . Uint64 ( ) ,
2020-11-14 14:30:45 +03:00
UserID : TestUserId ,
Project : TestProject1 ,
Language : TestLanguageGo ,
Editor : TestEditorGoland ,
OperatingSystem : TestOsLinux ,
Machine : TestMachine1 ,
Time : models . CustomTime ( suite . TestStartTime ) ,
} ,
{
2021-12-15 14:52:24 +03:00
ID : rand . Uint64 ( ) ,
2020-11-14 14:30:45 +03:00
UserID : TestUserId ,
Project : TestProject1 ,
Language : TestLanguageGo ,
Editor : TestEditorGoland ,
OperatingSystem : TestOsLinux ,
Machine : TestMachine1 ,
Time : models . CustomTime ( suite . TestStartTime . Add ( 30 * time . Second ) ) ,
} ,
{
2021-12-15 14:52:24 +03:00
ID : rand . Uint64 ( ) ,
2020-11-14 14:30:45 +03:00
UserID : TestUserId ,
Project : TestProject1 ,
Language : TestLanguageGo ,
Editor : TestEditorVscode ,
OperatingSystem : TestOsLinux ,
Machine : TestMachine1 ,
Time : models . CustomTime ( suite . TestStartTime . Add ( 3 * time . Minute ) ) ,
} ,
}
2021-06-12 12:26:15 +03:00
suite . TestLabels = [ ] * models . ProjectLabel {
{
ID : uint ( rand . Uint32 ( ) ) ,
UserID : TestUserId ,
ProjectKey : TestProject1 ,
Label : TestProjectLabel1 ,
} ,
{
ID : uint ( rand . Uint32 ( ) ) ,
UserID : TestUserId ,
ProjectKey : TestProjectLabel3 ,
Label : "blaahh" ,
} ,
}
2020-11-14 14:30:45 +03:00
}
func ( suite * SummaryServiceTestSuite ) BeforeTest ( suiteName , testName string ) {
suite . SummaryRepository = new ( mocks . SummaryRepositoryMock )
suite . HeartbeatService = new ( mocks . HeartbeatServiceMock )
suite . AliasService = new ( mocks . AliasServiceMock )
2021-06-12 12:09:24 +03:00
suite . ProjectLabelService = new ( mocks . ProjectLabelServiceMock )
2020-11-14 14:30:45 +03:00
}
func TestSummaryServiceTestSuite ( t * testing . T ) {
suite . Run ( t , new ( SummaryServiceTestSuite ) )
}
func ( suite * SummaryServiceTestSuite ) TestSummaryService_Summarize ( ) {
2021-06-12 12:09:24 +03:00
sut := NewSummaryService ( suite . SummaryRepository , suite . HeartbeatService , suite . AliasService , suite . ProjectLabelService )
2020-11-14 14:30:45 +03:00
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 ( ) {
2021-06-12 12:09:24 +03:00
sut := NewSummaryService ( suite . SummaryRepository , suite . HeartbeatService , suite . AliasService , suite . ProjectLabelService )
2020-11-14 14:30:45 +03:00
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 ) )
2021-04-12 23:57:15 +03:00
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 )
2020-11-14 14:30:45 +03:00
}
2021-04-19 21:48:07 +03:00
func ( suite * SummaryServiceTestSuite ) TestSummaryService_Retrieve_DuplicateSummaries ( ) {
2021-06-12 12:09:24 +03:00
sut := NewSummaryService ( suite . SummaryRepository , suite . HeartbeatService , suite . AliasService , suite . ProjectLabelService )
suite . ProjectLabelService . On ( "GetByUser" , suite . TestUser . ID ) . Return ( [ ] * models . ProjectLabel { } , nil )
2021-04-19 21:48:07 +03:00
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 )
}
2020-11-14 14:30:45 +03:00
func ( suite * SummaryServiceTestSuite ) TestSummaryService_Aliased ( ) {
2021-06-12 12:09:24 +03:00
sut := NewSummaryService ( suite . SummaryRepository , suite . HeartbeatService , suite . AliasService , suite . ProjectLabelService )
suite . ProjectLabelService . On ( "GetByUser" , suite . TestUser . ID ) . Return ( [ ] * models . ProjectLabel { } , nil )
2020-11-14 14:30:45 +03:00
var (
from time . Time
to time . Time
result * models . Summary
err error
)
from , to = suite . TestStartTime , suite . TestStartTime . Add ( 1 * time . Hour )
2021-08-07 01:12:45 +03:00
heartbeats := filter ( from , to , suite . TestHeartbeats )
heartbeats = append ( heartbeats , & models . Heartbeat {
2021-12-15 14:52:24 +03:00
ID : rand . Uint64 ( ) ,
2021-08-07 01:12:45 +03:00
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 )
2021-01-21 02:26:52 +03:00
suite . AliasService . On ( "InitializeUser" , TestUserId ) . Return ( nil )
2021-08-07 01:12:45 +03:00
suite . AliasService . On ( "GetAliasOrDefault" , TestUserId , mock . Anything , TestProject1 ) . Return ( TestProject2 , nil )
suite . AliasService . On ( "GetAliasOrDefault" , TestUserId , mock . Anything , TestProject2 ) . Return ( TestProject2 , nil )
2020-11-14 14:30:45 +03:00
suite . AliasService . On ( "GetAliasOrDefault" , TestUserId , mock . Anything , mock . Anything ) . Return ( "" , nil )
2021-08-07 01:12:45 +03:00
suite . ProjectLabelService . On ( "GetByUser" , suite . TestUser . ID ) . Return ( suite . TestLabels , nil ) . Once ( )
2020-11-14 14:30:45 +03:00
2021-03-25 01:31:04 +03:00
result , err = sut . Aliased ( from , to , suite . TestUser , sut . Summarize , false )
2020-11-14 14:30:45 +03:00
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 ) )
}
2021-08-07 01:12:45 +03:00
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 {
2021-12-15 14:52:24 +03:00
ID : rand . Uint64 ( ) ,
2021-08-07 01:12:45 +03:00
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 ) )
}
2020-11-14 14:30:45 +03:00
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 )
}
}