package handler_test import ( "os" "testing" "time" "github.com/golang/protobuf/ptypes/timestamp" "github.com/micro/services/threads/handler" pb "github.com/micro/services/threads/proto" "github.com/stretchr/testify/assert" "gorm.io/driver/postgres" "gorm.io/gorm" ) func testHandler(t *testing.T) *handler.Threads { // connect to the database addr := os.Getenv("POSTGRES_URL") if len(addr) == 0 { addr = "postgresql://postgres@localhost:5432/postgres?sslmode=disable" } db, err := gorm.Open(postgres.Open(addr), &gorm.Config{}) if err != nil { t.Fatalf("Error connecting to database: %v", err) } // clean any data from a previous run if err := db.Exec("DROP TABLE IF EXISTS conversations, messages CASCADE").Error; err != nil { t.Fatalf("Error cleaning database: %v", err) } // migrate the database if err := db.AutoMigrate(&handler.Conversation{}, &handler.Message{}); err != nil { t.Fatalf("Error migrating database: %v", err) } // clean any data from a previous run if err := db.Exec("TRUNCATE TABLE conversations, messages CASCADE").Error; err != nil { t.Fatalf("Error cleaning database: %v", err) } return &handler.Threads{DB: db, Time: func() time.Time { return time.Unix(1611327673, 0) }} } func assertConversationsMatch(t *testing.T, exp, act *pb.Conversation) { if act == nil { t.Errorf("Conversation not returned") return } // adapt this check so we can reuse the func in testing create, where we don't know the exact id // which will be generated if len(exp.Id) > 0 { assert.Equal(t, exp.Id, act.Id) } else { assert.NotEmpty(t, act.Id) } assert.Equal(t, exp.Topic, act.Topic) assert.Equal(t, exp.GroupId, act.GroupId) if act.CreatedAt == nil { t.Errorf("CreatedAt not set") return } assert.True(t, microSecondTime(exp.CreatedAt).Equal(microSecondTime(act.CreatedAt))) } func assertMessagesMatch(t *testing.T, exp, act *pb.Message) { if act == nil { t.Errorf("Message not returned") return } // adapt this check so we can reuse the func in testing create, where we don't know the exact id // which will be generated if len(exp.Id) > 0 { assert.Equal(t, exp.Id, act.Id) } else { assert.NotEmpty(t, act.Id) } assert.Equal(t, exp.Text, act.Text) assert.Equal(t, exp.AuthorId, act.AuthorId) assert.Equal(t, exp.ConversationId, act.ConversationId) if act.SentAt == nil { t.Errorf("SentAt not set") return } assert.True(t, microSecondTime(exp.SentAt).Equal(microSecondTime(act.SentAt))) } // postgres has a resolution of 100microseconds so just test that it's accurate to the second func microSecondTime(t *timestamp.Timestamp) time.Time { tt := t.AsTime() return time.Unix(tt.Unix(), int64(tt.Nanosecond()-tt.Nanosecond()%1000)) }