Compare commits
	
		
			24 Commits
		
	
	
		
			4d7589ad95
			...
			379d98ccd5
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					379d98ccd5 | ||
| f40e11c267 | |||
| b698f08136 | |||
| efdb52bccd | |||
| 2ff9fed3c5 | |||
| 477b4cf8f6 | |||
| ed6651697a | |||
| c359d99075 | |||
| 8d477356f3 | |||
| 7f3294a8a2 | |||
| 4d72ae0313 | |||
| 4fe64a7747 | |||
| 3004698d5b | |||
| 50c7ff1046 | |||
| 07b061a68b | |||
| 64fba91670 | |||
| c15958a1ad | |||
| ee442daefa | |||
| 11d30583a4 | |||
| f83f539046 | |||
| 562bd4fa24 | |||
| 90df0eefc3 | |||
| b37f2a0a11 | |||
| 68e4c9af03 | 
@ -55,6 +55,8 @@ const (
 | 
				
			|||||||
	DATA_ACTIVE_DECIMAL
 | 
						DATA_ACTIVE_DECIMAL
 | 
				
			||||||
	// EVM address of the currently active voucher
 | 
						// EVM address of the currently active voucher
 | 
				
			||||||
	DATA_ACTIVE_ADDRESS
 | 
						DATA_ACTIVE_ADDRESS
 | 
				
			||||||
 | 
						//Holds count of the number of incorrect PIN attempts
 | 
				
			||||||
 | 
						DATA_INCORRECT_PIN_ATTEMPTS
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const (
 | 
					const (
 | 
				
			||||||
 | 
				
			|||||||
@ -6,9 +6,13 @@ import (
 | 
				
			|||||||
	"golang.org/x/crypto/bcrypt"
 | 
						"golang.org/x/crypto/bcrypt"
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Define the regex pattern as a constant
 | 
					 | 
				
			||||||
const (
 | 
					const (
 | 
				
			||||||
 | 
						// Define the regex pattern as a constant
 | 
				
			||||||
	pinPattern = `^\d{4}$`
 | 
						pinPattern = `^\d{4}$`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						//Allowed incorrect  PIN attempts
 | 
				
			||||||
 | 
						AllowedPINAttempts = uint8(3)
 | 
				
			||||||
 | 
						
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// checks whether the given input is a 4 digit number
 | 
					// checks whether the given input is a 4 digit number
 | 
				
			||||||
 | 
				
			|||||||
@ -128,6 +128,7 @@ func (ls *LocalHandlerService) GetHandler(accountService remote.AccountServiceIn
 | 
				
			|||||||
	ls.DbRs.AddLocalFunc("view_statement", ussdHandlers.ViewTransactionStatement)
 | 
						ls.DbRs.AddLocalFunc("view_statement", ussdHandlers.ViewTransactionStatement)
 | 
				
			||||||
	ls.DbRs.AddLocalFunc("update_all_profile_items", ussdHandlers.UpdateAllProfileItems)
 | 
						ls.DbRs.AddLocalFunc("update_all_profile_items", ussdHandlers.UpdateAllProfileItems)
 | 
				
			||||||
	ls.DbRs.AddLocalFunc("set_back", ussdHandlers.SetBack)
 | 
						ls.DbRs.AddLocalFunc("set_back", ussdHandlers.SetBack)
 | 
				
			||||||
 | 
						ls.DbRs.AddLocalFunc("show_blocked_account", ussdHandlers.ShowBlockedAccount)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return ussdHandlers, nil
 | 
						return ussdHandlers, nil
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -734,11 +734,23 @@ func (h *Handlers) Authorize(ctx context.Context, sym string, input []byte) (res
 | 
				
			|||||||
			if h.st.MatchFlag(flag_account_authorized, false) {
 | 
								if h.st.MatchFlag(flag_account_authorized, false) {
 | 
				
			||||||
				res.FlagReset = append(res.FlagReset, flag_incorrect_pin)
 | 
									res.FlagReset = append(res.FlagReset, flag_incorrect_pin)
 | 
				
			||||||
				res.FlagSet = append(res.FlagSet, flag_allow_update, flag_account_authorized)
 | 
									res.FlagSet = append(res.FlagSet, flag_allow_update, flag_account_authorized)
 | 
				
			||||||
 | 
									err := h.resetIncorrectPINAttempts(ctx, sessionId)
 | 
				
			||||||
 | 
									if err != nil {
 | 
				
			||||||
 | 
										return res, err
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
			} else {
 | 
								} else {
 | 
				
			||||||
				res.FlagSet = append(res.FlagSet, flag_allow_update)
 | 
									res.FlagSet = append(res.FlagSet, flag_allow_update)
 | 
				
			||||||
				res.FlagReset = append(res.FlagReset, flag_account_authorized)
 | 
									res.FlagReset = append(res.FlagReset, flag_account_authorized)
 | 
				
			||||||
 | 
									err := h.resetIncorrectPINAttempts(ctx, sessionId)
 | 
				
			||||||
 | 
									if err != nil {
 | 
				
			||||||
 | 
										return res, err
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
		} else {
 | 
							} else {
 | 
				
			||||||
 | 
								err := h.incrementIncorrectPINAttempts(ctx, sessionId)
 | 
				
			||||||
 | 
								if err != nil {
 | 
				
			||||||
 | 
									return res, err
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
			res.FlagSet = append(res.FlagSet, flag_incorrect_pin)
 | 
								res.FlagSet = append(res.FlagSet, flag_incorrect_pin)
 | 
				
			||||||
			res.FlagReset = append(res.FlagReset, flag_account_authorized)
 | 
								res.FlagReset = append(res.FlagReset, flag_account_authorized)
 | 
				
			||||||
			return res, nil
 | 
								return res, nil
 | 
				
			||||||
@ -752,8 +764,34 @@ func (h *Handlers) Authorize(ctx context.Context, sym string, input []byte) (res
 | 
				
			|||||||
// ResetIncorrectPin resets the incorrect pin flag  after a new PIN attempt.
 | 
					// ResetIncorrectPin resets the incorrect pin flag  after a new PIN attempt.
 | 
				
			||||||
func (h *Handlers) ResetIncorrectPin(ctx context.Context, sym string, input []byte) (resource.Result, error) {
 | 
					func (h *Handlers) ResetIncorrectPin(ctx context.Context, sym string, input []byte) (resource.Result, error) {
 | 
				
			||||||
	var res resource.Result
 | 
						var res resource.Result
 | 
				
			||||||
 | 
						store := h.userdataStore
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	flag_incorrect_pin, _ := h.flagManager.GetFlag("flag_incorrect_pin")
 | 
						flag_incorrect_pin, _ := h.flagManager.GetFlag("flag_incorrect_pin")
 | 
				
			||||||
 | 
						flag_account_blocked, _ := h.flagManager.GetFlag("flag_account_blocked")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						sessionId, ok := ctx.Value("SessionId").(string)
 | 
				
			||||||
 | 
						if !ok {
 | 
				
			||||||
 | 
							return res, fmt.Errorf("missing session")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	res.FlagReset = append(res.FlagReset, flag_incorrect_pin)
 | 
						res.FlagReset = append(res.FlagReset, flag_incorrect_pin)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						currentWrongPinAttempts, err := store.ReadEntry(ctx, sessionId, common.DATA_INCORRECT_PIN_ATTEMPTS)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							if !db.IsNotFound(err) {
 | 
				
			||||||
 | 
								return res, err
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						pinAttemptsValue, _ := strconv.ParseUint(string(currentWrongPinAttempts), 0, 64)
 | 
				
			||||||
 | 
						remainingPINAttempts := common.AllowedPINAttempts - uint8(pinAttemptsValue)
 | 
				
			||||||
 | 
						if remainingPINAttempts == 0 {
 | 
				
			||||||
 | 
							res.FlagSet = append(res.FlagSet, flag_account_blocked)
 | 
				
			||||||
 | 
							return res, nil
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						if remainingPINAttempts < common.AllowedPINAttempts {
 | 
				
			||||||
 | 
							res.Content = strconv.Itoa(int(remainingPINAttempts))
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return res, nil
 | 
						return res, nil
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -840,6 +878,16 @@ func (h *Handlers) QuitWithHelp(ctx context.Context, sym string, input []byte) (
 | 
				
			|||||||
	return res, nil
 | 
						return res, nil
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ShowBlockedAccount displays a message after an account has been blocked and how to reach support.
 | 
				
			||||||
 | 
					func (h *Handlers) ShowBlockedAccount(ctx context.Context, sym string, input []byte) (resource.Result, error) {
 | 
				
			||||||
 | 
						var res resource.Result
 | 
				
			||||||
 | 
						code := codeFromCtx(ctx)
 | 
				
			||||||
 | 
						l := gotext.NewLocale(translationDir, code)
 | 
				
			||||||
 | 
						l.AddDomain("default")
 | 
				
			||||||
 | 
						res.Content = l.Get("Your account has been locked. For help on how to unblock your account, contact support at: 0757628885")
 | 
				
			||||||
 | 
						return res, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// VerifyYob verifies the length of the given input.
 | 
					// VerifyYob verifies the length of the given input.
 | 
				
			||||||
func (h *Handlers) VerifyYob(ctx context.Context, sym string, input []byte) (resource.Result, error) {
 | 
					func (h *Handlers) VerifyYob(ctx context.Context, sym string, input []byte) (resource.Result, error) {
 | 
				
			||||||
	var res resource.Result
 | 
						var res resource.Result
 | 
				
			||||||
@ -2075,3 +2123,53 @@ func (h *Handlers) UpdateAllProfileItems(ctx context.Context, sym string, input
 | 
				
			|||||||
	}
 | 
						}
 | 
				
			||||||
	return res, nil
 | 
						return res, nil
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// incrementIncorrectPINAttempts keeps track of the number of incorrect PIN attempts
 | 
				
			||||||
 | 
					func (h *Handlers) incrementIncorrectPINAttempts(ctx context.Context, sessionId string) error {
 | 
				
			||||||
 | 
						var pinAttemptsCount uint8
 | 
				
			||||||
 | 
						store := h.userdataStore
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						currentWrongPinAttempts, err := store.ReadEntry(ctx, sessionId, common.DATA_INCORRECT_PIN_ATTEMPTS)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							if db.IsNotFound(err) {
 | 
				
			||||||
 | 
								//First time Wrong PIN attempt: initialize with a count of 1
 | 
				
			||||||
 | 
								pinAttemptsCount = 1
 | 
				
			||||||
 | 
								err = store.WriteEntry(ctx, sessionId, common.DATA_INCORRECT_PIN_ATTEMPTS, []byte(strconv.Itoa(int(pinAttemptsCount))))
 | 
				
			||||||
 | 
								if err != nil {
 | 
				
			||||||
 | 
									logg.ErrorCtxf(ctx, "failed to write incorrect PIN attempts ", "key", common.DATA_INCORRECT_PIN_ATTEMPTS, "value", currentWrongPinAttempts, "error", err)
 | 
				
			||||||
 | 
									return err
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
								return nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						pinAttemptsValue, _ := strconv.ParseUint(string(currentWrongPinAttempts), 0, 64)
 | 
				
			||||||
 | 
						pinAttemptsCount = uint8(pinAttemptsValue) + 1
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						err = store.WriteEntry(ctx, sessionId, common.DATA_INCORRECT_PIN_ATTEMPTS, []byte(strconv.Itoa(int(pinAttemptsCount))))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							logg.ErrorCtxf(ctx, "failed to write incorrect PIN attempts ", "key", common.DATA_INCORRECT_PIN_ATTEMPTS, "value", pinAttemptsCount, "error", err)
 | 
				
			||||||
 | 
							return err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// resetIncorrectPINAttempts resets the number of incorrect PIN attempts after a correct PIN entry
 | 
				
			||||||
 | 
					func (h *Handlers) resetIncorrectPINAttempts(ctx context.Context, sessionId string) error {
 | 
				
			||||||
 | 
						store := h.userdataStore
 | 
				
			||||||
 | 
						currentWrongPinAttempts, err := store.ReadEntry(ctx, sessionId, common.DATA_INCORRECT_PIN_ATTEMPTS)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							if db.IsNotFound(err) {
 | 
				
			||||||
 | 
								return nil
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							return err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						currentWrongPinAttemptsCount, _ := strconv.ParseUint(string(currentWrongPinAttempts), 0, 64)
 | 
				
			||||||
 | 
						if currentWrongPinAttemptsCount <= uint64(common.AllowedPINAttempts) {
 | 
				
			||||||
 | 
							err = store.WriteEntry(ctx, sessionId, common.DATA_INCORRECT_PIN_ATTEMPTS, []byte(string("0")))
 | 
				
			||||||
 | 
							if err != nil {
 | 
				
			||||||
 | 
								logg.ErrorCtxf(ctx, "failed to reset incorrect PIN attempts ", "key", common.DATA_INCORRECT_PIN_ATTEMPTS, "value", common.AllowedPINAttempts, "error", err)
 | 
				
			||||||
 | 
								return err
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -5,6 +5,7 @@ import (
 | 
				
			|||||||
	"fmt"
 | 
						"fmt"
 | 
				
			||||||
	"log"
 | 
						"log"
 | 
				
			||||||
	"path"
 | 
						"path"
 | 
				
			||||||
 | 
						"strconv"
 | 
				
			||||||
	"strings"
 | 
						"strings"
 | 
				
			||||||
	"testing"
 | 
						"testing"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -907,37 +908,79 @@ func TestResetAccountAuthorized(t *testing.T) {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestIncorrectPinReset(t *testing.T) {
 | 
					func TestIncorrectPinReset(t *testing.T) {
 | 
				
			||||||
 | 
						sessionId := "session123"
 | 
				
			||||||
 | 
						ctx, store := InitializeTestStore(t)
 | 
				
			||||||
	fm, err := NewFlagManager(flagsPath)
 | 
						fm, err := NewFlagManager(flagsPath)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if err != nil {
 | 
						if err != nil {
 | 
				
			||||||
		log.Fatal(err)
 | 
							log.Fatal(err)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	flag_incorrect_pin, _ := fm.parser.GetFlag("flag_incorrect_pin")
 | 
						flag_incorrect_pin, _ := fm.parser.GetFlag("flag_incorrect_pin")
 | 
				
			||||||
 | 
						flag_account_blocked, _ := fm.parser.GetFlag("flag_account_blocked")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ctx = context.WithValue(ctx, "SessionId", sessionId)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	// Define test cases
 | 
						// Define test cases
 | 
				
			||||||
	tests := []struct {
 | 
						tests := []struct {
 | 
				
			||||||
		name           string
 | 
							name           string
 | 
				
			||||||
		input          []byte
 | 
							input          []byte
 | 
				
			||||||
 | 
							attempts       uint8
 | 
				
			||||||
		expectedResult resource.Result
 | 
							expectedResult resource.Result
 | 
				
			||||||
	}{
 | 
						}{
 | 
				
			||||||
		{
 | 
							{
 | 
				
			||||||
			name:  "Test incorrect pin reset",
 | 
								name:  "Test when incorrect PIN attempts is 2",
 | 
				
			||||||
			input: []byte(""),
 | 
								input: []byte(""),
 | 
				
			||||||
			expectedResult: resource.Result{
 | 
								expectedResult: resource.Result{
 | 
				
			||||||
				FlagReset: []uint32{flag_incorrect_pin},
 | 
									FlagReset: []uint32{flag_incorrect_pin},
 | 
				
			||||||
 | 
									Content:   "1", //Expected remaining PIN attempts
 | 
				
			||||||
			},
 | 
								},
 | 
				
			||||||
 | 
								attempts: 2,
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								name:  "Test incorrect pin reset when incorrect PIN attempts is 1",
 | 
				
			||||||
 | 
								input: []byte(""),
 | 
				
			||||||
 | 
								expectedResult: resource.Result{
 | 
				
			||||||
 | 
									FlagReset: []uint32{flag_incorrect_pin},
 | 
				
			||||||
 | 
									Content:   "2", //Expected remaining PIN attempts
 | 
				
			||||||
 | 
								},
 | 
				
			||||||
 | 
								attempts: 1,
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								name:  "Test incorrect pin reset when incorrect PIN attempts is 1",
 | 
				
			||||||
 | 
								input: []byte(""),
 | 
				
			||||||
 | 
								expectedResult: resource.Result{
 | 
				
			||||||
 | 
									FlagReset: []uint32{flag_incorrect_pin},
 | 
				
			||||||
 | 
									Content:   "2", //Expected remaining PIN attempts
 | 
				
			||||||
 | 
								},
 | 
				
			||||||
 | 
								attempts: 1,
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								name:  "Test incorrect pin reset when incorrect PIN attempts is 3(account expected to be blocked)",
 | 
				
			||||||
 | 
								input: []byte(""),
 | 
				
			||||||
 | 
								expectedResult: resource.Result{
 | 
				
			||||||
 | 
									FlagReset: []uint32{flag_incorrect_pin},
 | 
				
			||||||
 | 
									FlagSet:   []uint32{flag_account_blocked},
 | 
				
			||||||
 | 
								},
 | 
				
			||||||
 | 
								attempts: 3,
 | 
				
			||||||
		},
 | 
							},
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for _, tt := range tests {
 | 
						for _, tt := range tests {
 | 
				
			||||||
		t.Run(tt.name, func(t *testing.T) {
 | 
							t.Run(tt.name, func(t *testing.T) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								if err := store.WriteEntry(ctx, sessionId, common.DATA_INCORRECT_PIN_ATTEMPTS, []byte(strconv.Itoa(int(tt.attempts)))); err != nil {
 | 
				
			||||||
 | 
									t.Fatal(err)
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			// Create the Handlers instance with the mock flag manager
 | 
								// Create the Handlers instance with the mock flag manager
 | 
				
			||||||
			h := &Handlers{
 | 
								h := &Handlers{
 | 
				
			||||||
				flagManager: fm.parser,
 | 
									flagManager:   fm.parser,
 | 
				
			||||||
 | 
									userdataStore: store,
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			// Call the method
 | 
								// Call the method
 | 
				
			||||||
			res, err := h.ResetIncorrectPin(context.Background(), "reset_incorrect_pin", tt.input)
 | 
								res, err := h.ResetIncorrectPin(ctx, "reset_incorrect_pin", tt.input)
 | 
				
			||||||
			if err != nil {
 | 
								if err != nil {
 | 
				
			||||||
				t.Error(err)
 | 
									t.Error(err)
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
@ -2190,3 +2233,55 @@ func TestGetVoucherDetails(t *testing.T) {
 | 
				
			|||||||
	assert.NoError(t, err)
 | 
						assert.NoError(t, err)
 | 
				
			||||||
	assert.Equal(t, expectedResult, res)
 | 
						assert.Equal(t, expectedResult, res)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestCountIncorrectPINAttempts(t *testing.T) {
 | 
				
			||||||
 | 
						ctx, store := InitializeTestStore(t)
 | 
				
			||||||
 | 
						sessionId := "session123"
 | 
				
			||||||
 | 
						ctx = context.WithValue(ctx, "SessionId", sessionId)
 | 
				
			||||||
 | 
						attempts := uint8(2)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						h := &Handlers{
 | 
				
			||||||
 | 
							userdataStore: store,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						err := store.WriteEntry(ctx, sessionId, common.DATA_INCORRECT_PIN_ATTEMPTS, []byte(strconv.Itoa(int(attempts))))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							t.Logf(err.Error())
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						err = h.incrementIncorrectPINAttempts(ctx, sessionId)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							t.Logf(err.Error())
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						attemptsAfterCount, err := store.ReadEntry(ctx, sessionId, common.DATA_INCORRECT_PIN_ATTEMPTS)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							t.Logf(err.Error())
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						pinAttemptsValue, _ := strconv.ParseUint(string(attemptsAfterCount), 0, 64)
 | 
				
			||||||
 | 
						pinAttemptsCount := uint8(pinAttemptsValue)
 | 
				
			||||||
 | 
						expectedAttempts := attempts + 1
 | 
				
			||||||
 | 
						assert.Equal(t, pinAttemptsCount, expectedAttempts)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestResetIncorrectPINAttempts(t *testing.T) {
 | 
				
			||||||
 | 
						ctx, store := InitializeTestStore(t)
 | 
				
			||||||
 | 
						sessionId := "session123"
 | 
				
			||||||
 | 
						ctx = context.WithValue(ctx, "SessionId", sessionId)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						err := store.WriteEntry(ctx, sessionId, common.DATA_INCORRECT_PIN_ATTEMPTS, []byte(string("2")))
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							t.Logf(err.Error())
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						h := &Handlers{
 | 
				
			||||||
 | 
							userdataStore: store,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						h.resetIncorrectPINAttempts(ctx, sessionId)
 | 
				
			||||||
 | 
						incorrectAttempts, err := store.ReadEntry(ctx, sessionId, common.DATA_INCORRECT_PIN_ATTEMPTS)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							t.Logf(err.Error())
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						assert.Equal(t, "0", string(incorrectAttempts))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -54,7 +54,7 @@
 | 
				
			|||||||
                },
 | 
					                },
 | 
				
			||||||
                {
 | 
					                {
 | 
				
			||||||
                    "input": "1235",
 | 
					                    "input": "1235",
 | 
				
			||||||
                    "expectedContent": "Incorrect PIN\n1:Retry\n9:Quit"
 | 
					                    "expectedContent": "Incorrect PIN. You have: 2 remaining attempt(s).\n1:Retry\n9:Quit"
 | 
				
			||||||
                },
 | 
					                },
 | 
				
			||||||
                {
 | 
					                {
 | 
				
			||||||
                    "input": "1",
 | 
					                    "input": "1",
 | 
				
			||||||
@ -95,7 +95,7 @@
 | 
				
			|||||||
                },
 | 
					                },
 | 
				
			||||||
                {
 | 
					                {
 | 
				
			||||||
                    "input": "1235",
 | 
					                    "input": "1235",
 | 
				
			||||||
                    "expectedContent": "Incorrect PIN\n1:Retry\n9:Quit"
 | 
					                    "expectedContent": "Incorrect PIN. You have: 2 remaining attempt(s).\n1:Retry\n9:Quit"
 | 
				
			||||||
                },
 | 
					                },
 | 
				
			||||||
                {
 | 
					                {
 | 
				
			||||||
                    "input": "1",
 | 
					                    "input": "1",
 | 
				
			||||||
@ -107,8 +107,7 @@
 | 
				
			|||||||
                },
 | 
					                },
 | 
				
			||||||
                {
 | 
					                {
 | 
				
			||||||
                    "input": "0",
 | 
					                    "input": "0",
 | 
				
			||||||
                     "expectedContent": "Balances:\n1:My balance\n2:Community balance\n0:Back"
 | 
					                    "expectedContent": "Balances:\n1:My balance\n2:Community balance\n0:Back"
 | 
				
			||||||
                    
 | 
					 | 
				
			||||||
                },
 | 
					                },
 | 
				
			||||||
                {
 | 
					                {
 | 
				
			||||||
                    "input": "0",
 | 
					                    "input": "0",
 | 
				
			||||||
@ -141,7 +140,7 @@
 | 
				
			|||||||
                },
 | 
					                },
 | 
				
			||||||
                {
 | 
					                {
 | 
				
			||||||
                    "input": "1235",
 | 
					                    "input": "1235",
 | 
				
			||||||
                    "expectedContent": "Incorrect PIN\n1:Retry\n9:Quit"
 | 
					                    "expectedContent": "Incorrect PIN. You have: 2 remaining attempt(s).\n1:Retry\n9:Quit"
 | 
				
			||||||
                },
 | 
					                },
 | 
				
			||||||
                {
 | 
					                {
 | 
				
			||||||
                    "input": "1",
 | 
					                    "input": "1",
 | 
				
			||||||
@ -153,8 +152,7 @@
 | 
				
			|||||||
                },
 | 
					                },
 | 
				
			||||||
                {
 | 
					                {
 | 
				
			||||||
                    "input": "0",
 | 
					                    "input": "0",
 | 
				
			||||||
                     "expectedContent": "Balances:\n1:My balance\n2:Community balance\n0:Back"
 | 
					                    "expectedContent": "Balances:\n1:My balance\n2:Community balance\n0:Back"
 | 
				
			||||||
                    
 | 
					 | 
				
			||||||
                },
 | 
					                },
 | 
				
			||||||
                {
 | 
					                {
 | 
				
			||||||
                    "input": "0",
 | 
					                    "input": "0",
 | 
				
			||||||
@ -195,7 +193,7 @@
 | 
				
			|||||||
                },
 | 
					                },
 | 
				
			||||||
                {
 | 
					                {
 | 
				
			||||||
                    "input": "1",
 | 
					                    "input": "1",
 | 
				
			||||||
                    "expectedContent":  "Enter your year of birth\n0:Back"
 | 
					                    "expectedContent": "Enter your year of birth\n0:Back"
 | 
				
			||||||
                },
 | 
					                },
 | 
				
			||||||
                {
 | 
					                {
 | 
				
			||||||
                    "input": "1940",
 | 
					                    "input": "1940",
 | 
				
			||||||
@ -258,7 +256,6 @@
 | 
				
			|||||||
                    "input": "0",
 | 
					                    "input": "0",
 | 
				
			||||||
                    "expectedContent": "{balance}\n\n1:Send\n2:My Vouchers\n3:My Account\n4:Help\n9:Quit"
 | 
					                    "expectedContent": "{balance}\n\n1:Send\n2:My Vouchers\n3:My Account\n4:Help\n9:Quit"
 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
               
 | 
					 | 
				
			||||||
            ]
 | 
					            ]
 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
        {
 | 
					        {
 | 
				
			||||||
@ -444,9 +441,3 @@
 | 
				
			|||||||
        }
 | 
					        }
 | 
				
			||||||
    ]
 | 
					    ]
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
      
 | 
					 | 
				
			||||||
        
 | 
					 | 
				
			||||||
       
 | 
					 | 
				
			||||||
        
 | 
					 | 
				
			||||||
    
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										2
									
								
								services/registration/blocked_account.vis
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								services/registration/blocked_account.vis
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,2 @@
 | 
				
			|||||||
 | 
					LOAD show_blocked_account 0
 | 
				
			||||||
 | 
					HALT 
 | 
				
			||||||
@ -1 +1 @@
 | 
				
			|||||||
Incorrect PIN
 | 
					Incorrect PIN. You have: {{.reset_incorrect}} remaining attempt(s).
 | 
				
			||||||
@ -1,5 +1,7 @@
 | 
				
			|||||||
LOAD reset_incorrect 0
 | 
					LOAD reset_incorrect 0
 | 
				
			||||||
RELOAD reset_incorrect
 | 
					RELOAD reset_incorrect
 | 
				
			||||||
 | 
					MAP reset_incorrect
 | 
				
			||||||
 | 
					CATCH blocked_account flag_account_blocked 1
 | 
				
			||||||
MOUT retry 1
 | 
					MOUT retry 1
 | 
				
			||||||
MOUT quit 9
 | 
					MOUT quit 9
 | 
				
			||||||
HALT
 | 
					HALT
 | 
				
			||||||
 | 
				
			|||||||
@ -1 +1 @@
 | 
				
			|||||||
PIN ulioeka sio sahihi
 | 
					PIN ulioeka sio sahihi, una majaribio: {{.reset_incorrect}} yaliyobaki
 | 
				
			||||||
@ -10,6 +10,9 @@ msgstr "Asante kwa kutumia huduma ya Sarafu. Kwaheri!"
 | 
				
			|||||||
msgid "For more help, please call: 0757628885"
 | 
					msgid "For more help, please call: 0757628885"
 | 
				
			||||||
msgstr "Kwa usaidizi zaidi, piga: 0757628885"
 | 
					msgstr "Kwa usaidizi zaidi, piga: 0757628885"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					msgid "Your account has been locked. For help on how to unblock your account, contact support at: 0757628885"
 | 
				
			||||||
 | 
					msgstr "Akaunti yako imefungwa. Kwa usaidizi wa jinsi ya kufungua akaunti yako, wasiliana na usaidizi kwa: 0757628885"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
msgid "Balance: %s\n"
 | 
					msgid "Balance: %s\n"
 | 
				
			||||||
msgstr "Salio: %s\n"
 | 
					msgstr "Salio: %s\n"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -28,3 +28,5 @@ flag,flag_gender_set,34,this is set when the gender of the profile is set
 | 
				
			|||||||
flag,flag_location_set,35,this is set when the location of the profile is set
 | 
					flag,flag_location_set,35,this is set when the location of the profile is set
 | 
				
			||||||
flag,flag_offerings_set,36,this is set when the offerings of the profile is set
 | 
					flag,flag_offerings_set,36,this is set when the offerings of the profile is set
 | 
				
			||||||
flag,flag_back_set,37,this is set when it is a back navigation
 | 
					flag,flag_back_set,37,this is set when it is a back navigation
 | 
				
			||||||
 | 
					flag,flag_account_blocked,38,this is set when an account has been blocked after the allowed incorrect PIN attempts have been exceeded
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
		
		
			
  | 
@ -1,3 +1,4 @@
 | 
				
			|||||||
 | 
					CATCH blocked_account flag_account_blocked 1
 | 
				
			||||||
CATCH select_language flag_language_set 0
 | 
					CATCH select_language flag_language_set 0
 | 
				
			||||||
CATCH terms flag_account_created 0
 | 
					CATCH terms flag_account_created 0
 | 
				
			||||||
LOAD check_account_status 0
 | 
					LOAD check_account_status 0
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
		Reference in New Issue
	
	Block a user