ussd/test_engine/menu_traversal_test.go

604 lines
17 KiB
Go
Raw Normal View History

2024-09-27 14:02:29 +02:00
package main
import (
2024-09-30 12:21:17 +02:00
"bytes"
2024-09-27 14:02:29 +02:00
"context"
2024-10-04 16:04:33 +02:00
"log"
"math/rand"
"os"
2024-10-01 13:50:45 +02:00
"regexp"
2024-09-27 14:02:29 +02:00
"testing"
"time"
2024-09-27 14:02:29 +02:00
2024-09-30 12:21:17 +02:00
"git.grassecon.net/urdt/ussd/driver"
2024-09-30 17:05:20 +02:00
enginetest "git.grassecon.net/urdt/ussd/engine"
2024-10-03 16:47:48 +02:00
"github.com/gofrs/uuid"
2024-09-27 14:02:29 +02:00
)
2024-09-30 12:21:17 +02:00
var (
testData = driver.ReadData()
2024-10-04 16:04:33 +02:00
testStore = ".test_state"
sessionID string
2024-10-04 16:04:33 +02:00
src = rand.NewSource(42)
g = rand.New(src)
2024-09-30 12:21:17 +02:00
)
2024-09-27 14:02:29 +02:00
2024-10-04 16:04:33 +02:00
func GenerateSessionId() string {
uu := uuid.NewGenWithOptions(uuid.WithRandomReader(g))
v, err := uu.NewV4()
if err != nil {
2024-10-04 16:04:33 +02:00
panic(err)
}
2024-10-04 16:04:33 +02:00
return v.String()
}
2024-10-01 13:50:45 +02:00
// Extract the public key from the engine response
func extractPublicKey(response []byte) string {
// Regex pattern to match the public key starting with 0x and 40 characters
re := regexp.MustCompile(`0x[a-fA-F0-9]{40}`)
match := re.Find(response)
if match != nil {
return string(match)
}
return ""
}
func TestMain(m *testing.M) {
2024-10-04 16:04:33 +02:00
sessionID = GenerateSessionId()
testStore = ".test_state"
defer func() {
if err := os.RemoveAll(testStore); err != nil {
log.Fatalf("Failed to delete state store %s: %v", testStore, err)
}
}()
m.Run()
}
func TestAccountCreationSuccessful(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
defer fn()
2024-09-30 12:21:17 +02:00
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "account_creation_successful")
for _, group := range groups {
for _, step := range group.Steps {
2024-09-30 21:27:37 +02:00
cont, err := en.Exec(ctx, []byte(step.Input))
2024-09-30 17:05:20 +02:00
if err != nil {
t.Fatalf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
2024-09-30 21:27:37 +02:00
}
if !cont {
break
2024-09-30 12:21:17 +02:00
}
2024-09-30 17:05:20 +02:00
w := bytes.NewBuffer(nil)
_, err = en.Flush(ctx, w)
if err != nil {
t.Fatalf("Test case '%s' failed during Flush: %v", group.Name, err)
2024-09-30 17:05:20 +02:00
}
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
}
2024-09-30 12:21:17 +02:00
}
}
}
// Adding a sleep after the test to wait for registration to complete the process
time.Sleep(5 * time.Second)
2024-09-27 14:02:29 +02:00
}
2024-09-30 15:05:14 +02:00
func TestAccountRegistrationRejectTerms(t *testing.T) {
2024-10-03 16:47:48 +02:00
// Generate a new UUID for this edge case test
2024-10-04 16:04:33 +02:00
uu := uuid.NewGenWithOptions(uuid.WithRandomReader(g))
v, err := uu.NewV4()
if err != nil {
t.Fail()
}
edgeCaseSessionID := v.String()
en, fn := enginetest.TestEngine(edgeCaseSessionID)
defer fn()
2024-09-30 15:05:14 +02:00
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "account_creation_reject_terms")
for _, group := range groups {
for _, step := range group.Steps {
2024-09-30 21:12:49 +02:00
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Fatalf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
2024-09-30 21:12:49 +02:00
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Fatalf("Test case '%s' failed during Flush: %v", group.Name, err)
2024-09-30 21:12:49 +02:00
}
2024-09-30 15:05:14 +02:00
2024-09-30 21:12:49 +02:00
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
2024-09-30 15:05:14 +02:00
}
}
}
}
}
2024-10-01 12:15:05 +02:00
2024-10-01 12:26:47 +02:00
func TestSendWithInvalidInputs(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-01 12:15:05 +02:00
defer fn()
ctx := context.Background()
sessions := testData
for _, session := range sessions {
2024-10-01 12:26:47 +02:00
groups := driver.FilterGroupsByName(session.Groups, "send_with_invalid_inputs")
2024-10-01 12:15:05 +02:00
for _, group := range groups {
for _, step := range group.Steps {
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Fatalf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Fatalf("Test case '%s' failed during Flush: %v", group.Name, err)
}
b := w.Bytes()
2024-10-01 13:50:45 +02:00
// Extract the dynamic public key from the output
publicKey := extractPublicKey(b)
// Replace placeholder {public_key} with the actual dynamic public key
expectedContent := bytes.Replace([]byte(step.ExpectedContent), []byte("{public_key}"), []byte(publicKey), -1)
if !bytes.Equal(b, expectedContent) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", expectedContent, b)
2024-10-01 12:15:05 +02:00
}
}
}
}
}
2024-10-01 14:36:59 +02:00
func TestMyAccount_Check_My_Balance(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-01 14:36:59 +02:00
defer fn()
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "menu_my_account_check_my_balance")
2024-10-01 14:36:59 +02:00
for _, group := range groups {
for index, step := range group.Steps {
t.Logf("step %v with input %v", index, step.Input)
2024-10-01 14:36:59 +02:00
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Errorf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
2024-10-01 14:36:59 +02:00
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Errorf("Test case '%s' failed during Flush: %v", group.Name, err)
2024-10-01 14:36:59 +02:00
}
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
}
}
}
}
}
func TestMainMenuHelp(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-01 14:36:59 +02:00
defer fn()
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "main_menu_help")
2024-10-01 14:36:59 +02:00
for _, group := range groups {
for _, step := range group.Steps {
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Fatalf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Fatalf("Test case '%s' failed during Flush: %v", group.Name, err)
}
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
}
}
}
}
}
2024-10-01 20:43:52 +02:00
func TestMainMenuQuit(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-01 20:43:52 +02:00
defer fn()
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "main_menu_quit")
2024-10-01 20:43:52 +02:00
for _, group := range groups {
for _, step := range group.Steps {
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Fatalf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Fatalf("Test case '%s' failed during Flush: %v", group.Name, err)
}
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
2024-09-30 21:12:49 +02:00
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
2024-09-30 15:05:14 +02:00
}
}
}
}
}
2024-10-01 15:54:16 +02:00
func TestMyAccount_Check_Community_Balance(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-01 22:21:29 +02:00
defer fn()
2024-10-01 15:54:16 +02:00
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "menu_my_account_check_community_balance")
2024-10-01 15:54:16 +02:00
for _, group := range groups {
for index, step := range group.Steps {
t.Logf("step %v with input %v", index, step.Input)
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Errorf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Errorf("Test case '%s' failed during Flush: %v", group.Name, err)
}
2024-10-02 08:55:56 +02:00
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
}
}
}
}
}
2024-10-01 15:54:16 +02:00
func TestMyAccountChangePin(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-02 08:55:56 +02:00
defer fn()
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "my_account_change_pin")
2024-10-02 08:55:56 +02:00
for _, group := range groups {
for _, step := range group.Steps {
2024-10-02 08:55:56 +02:00
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Fatalf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
2024-10-02 08:55:56 +02:00
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Fatalf("Test case '%s' failed during Flush: %v", group.Name, err)
2024-10-02 08:55:56 +02:00
}
2024-10-01 15:54:16 +02:00
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
}
}
}
}
}
func TestMyAccount_Change_Language(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-01 22:21:29 +02:00
defer fn()
2024-10-01 15:54:16 +02:00
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "menu_my_account_language_change")
2024-10-01 15:54:16 +02:00
for _, group := range groups {
for index, step := range group.Steps {
t.Logf("step %v with input %v", index, step.Input)
2024-10-02 08:55:56 +02:00
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Errorf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Errorf("Test case '%s' failed during Flush: %v", group.Name, err)
}
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
}
}
}
}
}
func TestMyAccount_Edit_firstname(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-02 08:55:56 +02:00
defer fn()
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "menu_my_account_edit_firstname")
2024-10-02 08:55:56 +02:00
for _, group := range groups {
for index, step := range group.Steps {
t.Logf("step %v with input %v", index, step.Input)
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Errorf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Errorf("Test case '%s' failed during Flush: %v", group.Name, err)
}
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
}
}
}
}
}
func TestMyAccount_Edit_familyname(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-02 08:55:56 +02:00
defer fn()
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "menu_my_account_edit_familyname")
2024-10-02 08:55:56 +02:00
for _, group := range groups {
for index, step := range group.Steps {
t.Logf("step %v with input %v", index, step.Input)
cont, err := en.Exec(ctx, []byte(step.Input))
2024-10-01 15:54:16 +02:00
if err != nil {
t.Errorf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
return
}
2024-10-02 08:55:56 +02:00
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Errorf("Test case '%s' failed during Flush: %v", group.Name, err)
}
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
}
}
}
}
}
func TestMyAccount_Edit_gender(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-02 08:55:56 +02:00
defer fn()
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "menu_my_account_edit_gender")
2024-10-02 08:55:56 +02:00
for _, group := range groups {
for index, step := range group.Steps {
t.Logf("step %v with input %v", index, step.Input)
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Errorf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Errorf("Test case '%s' failed during Flush: %v", group.Name, err)
}
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
}
}
}
}
}
func TestMyAccount_Edit_yob(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-02 08:55:56 +02:00
defer fn()
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "menu_my_account_edit_yob")
2024-10-02 08:55:56 +02:00
for _, group := range groups {
for index, step := range group.Steps {
t.Logf("step %v with input %v", index, step.Input)
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Errorf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
return
}
if !cont {
break
}
2024-10-01 15:54:16 +02:00
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Errorf("Test case '%s' failed during Flush: %v", group.Name, err)
}
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
}
}
}
}
}
2024-10-02 09:18:04 +02:00
func TestMyAccount_Edit_location(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-02 09:18:04 +02:00
defer fn()
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "menu_my_account_edit_location")
2024-10-02 09:18:04 +02:00
for _, group := range groups {
for index, step := range group.Steps {
t.Logf("step %v with input %v", index, step.Input)
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Errorf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Errorf("Test case '%s' failed during Flush: %v", group.Name, err)
}
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
}
}
}
}
}
2024-10-02 10:23:34 +02:00
func TestMyAccount_Edit_offerings(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-02 12:22:00 +02:00
defer fn()
ctx := context.Background()
sessions := testData
for _, session := range sessions {
groups := driver.FilterGroupsByName(session.Groups, "menu_my_account_edit_offerings")
2024-10-02 12:22:00 +02:00
for _, group := range groups {
for index, step := range group.Steps {
t.Logf("step %v with input %v", index, step.Input)
2024-10-04 10:11:41 +02:00
2024-10-02 12:22:00 +02:00
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Errorf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Errorf("Test case '%s' failed during Flush: %v", group.Name, err)
}
b := w.Bytes()
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
}
}
}
}
}
2024-10-04 10:11:41 +02:00
func TestMyAccount_MyAddress(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-02 10:23:34 +02:00
defer fn()
ctx := context.Background()
sessions := testData
for _, session := range sessions {
2024-10-04 10:11:41 +02:00
groups := driver.FilterGroupsByName(session.Groups, "menu_my_account_my_address")
2024-10-02 10:23:34 +02:00
for _, group := range groups {
for index, step := range group.Steps {
t.Logf("step %v with input %v", index, step.Input)
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Errorf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Errorf("Test case '%s' failed during Flush: %v", group.Name, err)
}
b := w.Bytes()
2024-10-04 10:11:41 +02:00
publicKey := extractPublicKey(b)
expectedContent := bytes.Replace([]byte(step.ExpectedContent), []byte("{public_key}"), []byte(publicKey), -1)
if !bytes.Equal(b, expectedContent) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", expectedContent, b)
2024-10-02 10:23:34 +02:00
}
}
}
}
}
2024-10-04 10:11:41 +02:00
func TestMyAccount_View_Profile(t *testing.T) {
2024-10-04 16:04:33 +02:00
en, fn := enginetest.TestEngine(sessionID)
2024-10-02 11:10:32 +02:00
defer fn()
ctx := context.Background()
sessions := testData
for _, session := range sessions {
2024-10-04 10:11:41 +02:00
groups := driver.FilterGroupsByName(session.Groups, "menu_my_account_view_profile")
2024-10-02 11:10:32 +02:00
for _, group := range groups {
for index, step := range group.Steps {
t.Logf("step %v with input %v", index, step.Input)
cont, err := en.Exec(ctx, []byte(step.Input))
if err != nil {
t.Errorf("Test case '%s' failed at input '%s': %v", group.Name, step.Input, err)
return
}
if !cont {
break
}
w := bytes.NewBuffer(nil)
if _, err := en.Flush(ctx, w); err != nil {
t.Errorf("Test case '%s' failed during Flush: %v", group.Name, err)
}
b := w.Bytes()
2024-10-04 10:11:41 +02:00
if !bytes.Equal(b, []byte(step.ExpectedContent)) {
t.Fatalf("expected:\n\t%s\ngot:\n\t%s\n", step.ExpectedContent, b)
2024-10-02 14:50:52 +02:00
}
2024-10-02 11:10:32 +02:00
}
}
}
}