Compare commits
No commits in common. "master" and "wip-flag-migration" have entirely different histories.
master
...
wip-flag-m
20
.env.example
20
.env.example
@ -1,20 +0,0 @@
|
|||||||
#Serve Http
|
|
||||||
PORT=7123
|
|
||||||
HOST=127.0.0.1
|
|
||||||
|
|
||||||
#AfricasTalking USSD POST endpoint
|
|
||||||
AT_ENDPOINT=/ussd/africastalking
|
|
||||||
|
|
||||||
#PostgreSQL
|
|
||||||
DB_CONN=postgres://postgres:strongpass@localhost:5432/urdt_ussd
|
|
||||||
#DB_TIMEZONE=Africa/Nairobi
|
|
||||||
#DB_SCHEMA=vise
|
|
||||||
|
|
||||||
#External API Calls
|
|
||||||
CUSTODIAL_URL_BASE=http://localhost:5003
|
|
||||||
BEARER_TOKEN=eyJeSIsInRcCI6IkpXVCJ.yJwdWJsaWNLZXkiOiIwrrrrrr
|
|
||||||
DATA_URL_BASE=http://localhost:5006
|
|
||||||
|
|
||||||
#Language
|
|
||||||
DEFAULT_LANGUAGE=eng
|
|
||||||
LANGUAGES=eng, swa
|
|
3
.gitignore
vendored
3
.gitignore
vendored
@ -4,6 +4,3 @@ go.work*
|
|||||||
**/*/*.bin
|
**/*/*.bin
|
||||||
**/*/.state/
|
**/*/.state/
|
||||||
cmd/.state/
|
cmd/.state/
|
||||||
id_*
|
|
||||||
*.gdbm
|
|
||||||
*.log
|
|
||||||
|
9
README.md
Normal file
9
README.md
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
# ussd
|
||||||
|
|
||||||
|
> USSD
|
||||||
|
|
||||||
|
USSD service.
|
||||||
|
|
||||||
|
## License
|
||||||
|
|
||||||
|
[AGPL-3.0](LICENSE).
|
172
cmd/main.go
Normal file
172
cmd/main.go
Normal file
@ -0,0 +1,172 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"encoding/csv"
|
||||||
|
"flag"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"os"
|
||||||
|
"path"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"git.defalsify.org/vise.git/cache"
|
||||||
|
"git.defalsify.org/vise.git/engine"
|
||||||
|
"git.defalsify.org/vise.git/persist"
|
||||||
|
"git.defalsify.org/vise.git/resource"
|
||||||
|
"git.defalsify.org/vise.git/state"
|
||||||
|
"git.grassecon.net/urdt/ussd/internal/handlers/ussd"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
scriptDir = path.Join("services", "registration")
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
var dir string
|
||||||
|
var root string
|
||||||
|
var size uint
|
||||||
|
var sessionId string
|
||||||
|
flag.StringVar(&dir, "d", ".", "resource dir to read from")
|
||||||
|
flag.UintVar(&size, "s", 0, "max size of output")
|
||||||
|
flag.StringVar(&root, "root", "root", "entry point symbol")
|
||||||
|
flag.StringVar(&sessionId, "session-id", "default", "session id")
|
||||||
|
flag.Parse()
|
||||||
|
fmt.Fprintf(os.Stderr, "starting session at symbol '%s' using resource dir: %s\n", root, dir)
|
||||||
|
|
||||||
|
ctx := context.Background()
|
||||||
|
st := state.NewState(16)
|
||||||
|
st.UseDebug()
|
||||||
|
|
||||||
|
pfp := path.Join(scriptDir, "pp.csv")
|
||||||
|
file, err := os.Open(pfp)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Failed to open CSV file: %v\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
reader := csv.NewReader(file)
|
||||||
|
|
||||||
|
// Iterate through the CSV records and register the flags
|
||||||
|
for {
|
||||||
|
record, err := reader.Read()
|
||||||
|
if err != nil {
|
||||||
|
if err == io.EOF {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
fmt.Fprintf(os.Stderr, "Error reading CSV file: %v\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure the record starts with "flag" and has at least 3 columns
|
||||||
|
if len(record) < 3 || record[0] != "flag" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
flagName := record[1]
|
||||||
|
flagValue, err := strconv.Atoi(record[2])
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Failed to convert flag value %s to integer: %v\n", record[2], err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Register the flag
|
||||||
|
state.FlagDebugger.Register(uint32(flagValue), flagName)
|
||||||
|
}
|
||||||
|
|
||||||
|
rfs := resource.NewFsResource(scriptDir)
|
||||||
|
ca := cache.NewCache()
|
||||||
|
cfg := engine.Config{
|
||||||
|
Root: "root",
|
||||||
|
SessionId: sessionId,
|
||||||
|
}
|
||||||
|
|
||||||
|
dp := path.Join(scriptDir, ".state")
|
||||||
|
err = os.MkdirAll(dp, 0700)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "state dir create exited with error: %v\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
pr := persist.NewFsPersister(dp)
|
||||||
|
en, err := engine.NewPersistedEngine(ctx, cfg, pr, rfs)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
pr = pr.WithContent(&st, ca)
|
||||||
|
err = pr.Save(cfg.SessionId)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "Failed to save state with error: %v\n", err)
|
||||||
|
}
|
||||||
|
en, err = engine.NewPersistedEngine(ctx, cfg, pr, rfs)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "engine create exited with error: %v\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fp := path.Join(dp, sessionId)
|
||||||
|
|
||||||
|
ussdHandlers, err := ussd.NewHandlers(fp, &st,sessionId)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "handler setup failed with error: %v\n", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
rfs.AddLocalFunc("select_language", ussdHandlers.SetLanguage)
|
||||||
|
rfs.AddLocalFunc("create_account", ussdHandlers.CreateAccount)
|
||||||
|
rfs.AddLocalFunc("save_pin", ussdHandlers.SavePin)
|
||||||
|
rfs.AddLocalFunc("verify_pin", ussdHandlers.VerifyPin)
|
||||||
|
rfs.AddLocalFunc("check_identifier", ussdHandlers.CheckIdentifier)
|
||||||
|
rfs.AddLocalFunc("check_account_status", ussdHandlers.CheckAccountStatus)
|
||||||
|
rfs.AddLocalFunc("authorize_account", ussdHandlers.Authorize)
|
||||||
|
rfs.AddLocalFunc("quit", ussdHandlers.Quit)
|
||||||
|
rfs.AddLocalFunc("check_balance", ussdHandlers.CheckBalance)
|
||||||
|
rfs.AddLocalFunc("validate_recipient", ussdHandlers.ValidateRecipient)
|
||||||
|
rfs.AddLocalFunc("transaction_reset", ussdHandlers.TransactionReset)
|
||||||
|
rfs.AddLocalFunc("max_amount", ussdHandlers.MaxAmount)
|
||||||
|
rfs.AddLocalFunc("validate_amount", ussdHandlers.ValidateAmount)
|
||||||
|
rfs.AddLocalFunc("reset_transaction_amount", ussdHandlers.ResetTransactionAmount)
|
||||||
|
rfs.AddLocalFunc("get_recipient", ussdHandlers.GetRecipient)
|
||||||
|
rfs.AddLocalFunc("get_sender", ussdHandlers.GetSender)
|
||||||
|
rfs.AddLocalFunc("get_amount", ussdHandlers.GetAmount)
|
||||||
|
rfs.AddLocalFunc("reset_incorrect", ussdHandlers.ResetIncorrectPin)
|
||||||
|
rfs.AddLocalFunc("save_firstname", ussdHandlers.SaveFirstname)
|
||||||
|
rfs.AddLocalFunc("save_familyname", ussdHandlers.SaveFamilyname)
|
||||||
|
rfs.AddLocalFunc("save_gender", ussdHandlers.SaveGender)
|
||||||
|
rfs.AddLocalFunc("save_location", ussdHandlers.SaveLocation)
|
||||||
|
rfs.AddLocalFunc("save_yob", ussdHandlers.SaveYob)
|
||||||
|
rfs.AddLocalFunc("save_offerings", ussdHandlers.SaveOfferings)
|
||||||
|
rfs.AddLocalFunc("quit_with_balance", ussdHandlers.QuitWithBalance)
|
||||||
|
rfs.AddLocalFunc("reset_account_authorized", ussdHandlers.ResetAccountAuthorized)
|
||||||
|
rfs.AddLocalFunc("reset_allow_update", ussdHandlers.ResetAllowUpdate)
|
||||||
|
rfs.AddLocalFunc("get_profile_info", ussdHandlers.GetProfileInfo)
|
||||||
|
rfs.AddLocalFunc("verify_yob", ussdHandlers.VerifyYob)
|
||||||
|
rfs.AddLocalFunc("reset_incorrect_date_format", ussdHandlers.ResetIncorrectYob)
|
||||||
|
rfs.AddLocalFunc("set_reset_single_edit", ussdHandlers.SetResetSingleEdit)
|
||||||
|
rfs.AddLocalFunc("initiate_transaction", ussdHandlers.InitiateTransaction)
|
||||||
|
|
||||||
|
cont, err := en.Init(ctx)
|
||||||
|
en.SetDebugger(engine.NewSimpleDebug(nil))
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "engine init exited with error: %v\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
if !cont {
|
||||||
|
_, err = en.WriteResult(ctx, os.Stdout)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "dead init write error: %v\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
err = en.Finish()
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "engine finish error: %v\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
os.Stdout.Write([]byte{0x0a})
|
||||||
|
os.Exit(0)
|
||||||
|
}
|
||||||
|
err = engine.Loop(ctx, en, os.Stdin, os.Stdout)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "loop exited with error: %v\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
149
config/config.go
149
config/config.go
@ -1,149 +1,10 @@
|
|||||||
package config
|
package config
|
||||||
|
|
||||||
import (
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"git.defalsify.org/vise.git/logging"
|
|
||||||
"git.grassecon.net/grassrootseconomics/visedriver/env"
|
const (
|
||||||
"git.grassecon.net/grassrootseconomics/visedriver/storage"
|
CreateAccountURL = "https://custodial.sarafu.africa/api/account/create"
|
||||||
|
TrackStatusURL = "https://custodial.sarafu.africa/api/track/"
|
||||||
|
BalanceURL = "https://custodial.sarafu.africa/api/account/status/"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
|
||||||
logg = logging.NewVanilla().WithDomain("visedriver-config")
|
|
||||||
defaultLanguage = "eng"
|
|
||||||
languages []string
|
|
||||||
DefaultLanguage string
|
|
||||||
dbConn string
|
|
||||||
dbConnMissing bool
|
|
||||||
dbConnMode storage.DbMode
|
|
||||||
stateDbConn string
|
|
||||||
stateDbConnMode storage.DbMode
|
|
||||||
resourceDbConn string
|
|
||||||
resourceDbConnMode storage.DbMode
|
|
||||||
userDbConn string
|
|
||||||
userDbConnMode storage.DbMode
|
|
||||||
Languages []string
|
|
||||||
)
|
|
||||||
|
|
||||||
type Override struct {
|
|
||||||
DbConn string
|
|
||||||
DbConnMode storage.DbMode
|
|
||||||
StateConn string
|
|
||||||
StateConnMode storage.DbMode
|
|
||||||
ResourceConn string
|
|
||||||
ResourceConnMode storage.DbMode
|
|
||||||
UserConn string
|
|
||||||
UserConnMode storage.DbMode
|
|
||||||
}
|
|
||||||
|
|
||||||
func setLanguage() error {
|
|
||||||
defaultLanguage = env.GetEnv("DEFAULT_LANGUAGE", defaultLanguage)
|
|
||||||
languages = strings.Split(env.GetEnv("LANGUAGES", defaultLanguage), ",")
|
|
||||||
haveDefaultLanguage := false
|
|
||||||
for i, v := range languages {
|
|
||||||
languages[i] = strings.ReplaceAll(v, " ", "")
|
|
||||||
if languages[i] == defaultLanguage {
|
|
||||||
haveDefaultLanguage = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if !haveDefaultLanguage {
|
|
||||||
languages = append([]string{defaultLanguage}, languages...)
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func setConn() error {
|
|
||||||
dbConn = env.GetEnv("DB_CONN", "?")
|
|
||||||
stateDbConn = env.GetEnv("DB_CONN_STATE", dbConn)
|
|
||||||
resourceDbConn = env.GetEnv("DB_CONN_RESOURCE", dbConn)
|
|
||||||
userDbConn = env.GetEnv("DB_CONN_USER", dbConn)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func ApplyConn(override *Override) {
|
|
||||||
if override.DbConn != "?" {
|
|
||||||
dbConn = override.DbConn
|
|
||||||
stateDbConn = override.StateConn
|
|
||||||
resourceDbConn = override.ResourceConn
|
|
||||||
userDbConn = override.UserConn
|
|
||||||
}
|
|
||||||
dbConnMode = override.DbConnMode
|
|
||||||
if override.StateConn != "?" {
|
|
||||||
stateDbConn = override.StateConn
|
|
||||||
}
|
|
||||||
if override.ResourceConn != "?" {
|
|
||||||
resourceDbConn = override.ResourceConn
|
|
||||||
}
|
|
||||||
if override.UserConn != "?" {
|
|
||||||
userDbConn = override.UserConn
|
|
||||||
}
|
|
||||||
|
|
||||||
if dbConn == "?" {
|
|
||||||
dbConn = ""
|
|
||||||
}
|
|
||||||
|
|
||||||
if stateDbConn == "?" {
|
|
||||||
stateDbConn = dbConn
|
|
||||||
stateDbConnMode = dbConnMode
|
|
||||||
}
|
|
||||||
if resourceDbConn == "?" {
|
|
||||||
resourceDbConn = dbConn
|
|
||||||
resourceDbConnMode = dbConnMode
|
|
||||||
}
|
|
||||||
if userDbConn == "?" {
|
|
||||||
userDbConn = dbConn
|
|
||||||
userDbConnMode = dbConnMode
|
|
||||||
}
|
|
||||||
|
|
||||||
logg.Debugf("conns", "conn", dbConn, "user", userDbConn)
|
|
||||||
if override.DbConnMode != storage.DBMODE_ANY {
|
|
||||||
dbConnMode = override.DbConnMode
|
|
||||||
}
|
|
||||||
if override.StateConnMode != storage.DBMODE_ANY {
|
|
||||||
stateDbConnMode = override.StateConnMode
|
|
||||||
}
|
|
||||||
if override.ResourceConnMode != storage.DBMODE_ANY {
|
|
||||||
resourceDbConnMode = override.ResourceConnMode
|
|
||||||
}
|
|
||||||
if override.UserConnMode != storage.DBMODE_ANY {
|
|
||||||
userDbConnMode = override.UserConnMode
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetConns() (storage.Conns, error) {
|
|
||||||
o := storage.NewConns()
|
|
||||||
c, err := storage.ToConnDataMode(stateDbConn, stateDbConnMode)
|
|
||||||
if err != nil {
|
|
||||||
return o, err
|
|
||||||
}
|
|
||||||
o.Set(c, storage.STORETYPE_STATE)
|
|
||||||
c, err = storage.ToConnDataMode(resourceDbConn, resourceDbConnMode)
|
|
||||||
if err != nil {
|
|
||||||
return o, err
|
|
||||||
}
|
|
||||||
o.Set(c, storage.STORETYPE_RESOURCE)
|
|
||||||
c, err = storage.ToConnDataMode(userDbConn, userDbConnMode)
|
|
||||||
if err != nil {
|
|
||||||
return o, err
|
|
||||||
}
|
|
||||||
o.Set(c, storage.STORETYPE_USER)
|
|
||||||
return o, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// LoadConfig initializes the configuration values after environment variables are loaded.
|
|
||||||
func LoadConfig() error {
|
|
||||||
err := setConn()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
err = setLanguage()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
DefaultLanguage = defaultLanguage
|
|
||||||
Languages = languages
|
|
||||||
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
28
doc/data.md
28
doc/data.md
@ -1,28 +0,0 @@
|
|||||||
# Internals
|
|
||||||
|
|
||||||
## Version
|
|
||||||
|
|
||||||
This document describes component versions:
|
|
||||||
|
|
||||||
* `urdt-ussd` `v0.5.0-beta`
|
|
||||||
* `go-vise` `v0.2.2`
|
|
||||||
|
|
||||||
|
|
||||||
## User profile data
|
|
||||||
|
|
||||||
All user profile items are stored under keys matching the user's session id, prefixed with the 8-bit value `git.defalsify.org/vise.git/db.DATATYPE_USERDATA` (32), and followed with a 16-big big-endian value subprefix.
|
|
||||||
|
|
||||||
For example, given the sessionId `+254123` and the key `git.grassecon.net/urdt-ussd/common.DATA_PUBLIC_KEY` (2) will be stored under the key:
|
|
||||||
|
|
||||||
```
|
|
||||||
0x322b3235343132330002
|
|
||||||
|
|
||||||
prefix sessionid subprefix
|
|
||||||
32 2b323534313233 0002
|
|
||||||
```
|
|
||||||
|
|
||||||
### Sub-prefixes
|
|
||||||
|
|
||||||
All sub-prefixes are defined as constants in the `git.grassecon.net/urdt-ussd/common` package. The constant names have the prefix `DATA_`
|
|
||||||
|
|
||||||
Please refer to inline godoc documentation for the `git.grassecon.net/urdt-ussd/common` package for details on each data item.
|
|
@ -1,14 +0,0 @@
|
|||||||
package entry
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
|
|
||||||
"git.defalsify.org/vise.git/persist"
|
|
||||||
"git.defalsify.org/vise.git/resource"
|
|
||||||
)
|
|
||||||
|
|
||||||
type EntryHandler interface {
|
|
||||||
Init(context.Context, string, []byte) (resource.Result, error) // HandlerFunc
|
|
||||||
Exit()
|
|
||||||
SetPersister(*persist.Persister)
|
|
||||||
}
|
|
40
env/load.go
vendored
40
env/load.go
vendored
@ -1,40 +0,0 @@
|
|||||||
package env
|
|
||||||
|
|
||||||
import (
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
"path"
|
|
||||||
"strconv"
|
|
||||||
|
|
||||||
"github.com/joho/godotenv"
|
|
||||||
)
|
|
||||||
|
|
||||||
func LoadEnvVariables() {
|
|
||||||
LoadEnvVariablesPath(".")
|
|
||||||
}
|
|
||||||
|
|
||||||
func LoadEnvVariablesPath(dir string) {
|
|
||||||
fp := path.Join(dir, ".env")
|
|
||||||
err := godotenv.Load(fp)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatal("Error loading .env file", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Helper to get environment variables with a default fallback
|
|
||||||
func GetEnv(key, defaultVal string) string {
|
|
||||||
if value, exists := os.LookupEnv(key); exists {
|
|
||||||
return value
|
|
||||||
}
|
|
||||||
return defaultVal
|
|
||||||
}
|
|
||||||
|
|
||||||
// Helper to safely convert environment variables to uint
|
|
||||||
func GetEnvUint(key string, defaultVal uint) uint {
|
|
||||||
if value, exists := os.LookupEnv(key); exists {
|
|
||||||
if parsed, err := strconv.Atoi(value); err == nil && parsed >= 0 {
|
|
||||||
return uint(parsed)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return defaultVal
|
|
||||||
}
|
|
@ -1,15 +0,0 @@
|
|||||||
package errors
|
|
||||||
|
|
||||||
import (
|
|
||||||
"errors"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
ErrInvalidRequest = errors.New("invalid request for context")
|
|
||||||
ErrSessionMissing = errors.New("missing session")
|
|
||||||
ErrInvalidInput = errors.New("invalid input")
|
|
||||||
ErrStorage = errors.New("storage retrieval fail")
|
|
||||||
ErrEngineType = errors.New("incompatible engine")
|
|
||||||
ErrEngineInit = errors.New("engine init fail")
|
|
||||||
ErrEngineExec = errors.New("engine exec fail")
|
|
||||||
)
|
|
1
go-vise
Submodule
1
go-vise
Submodule
@ -0,0 +1 @@
|
|||||||
|
Subproject commit 1f47a674d95380be8c387f410f0342eb72357df5
|
28
go.mod
28
go.mod
@ -1,27 +1,5 @@
|
|||||||
module git.grassecon.net/grassrootseconomics/visedriver
|
module git.grassecon.net/urdt/ussd
|
||||||
|
|
||||||
go 1.23.0
|
go 1.22.6
|
||||||
|
|
||||||
require (
|
require github.com/stretchr/testify v1.9.0 // indirect
|
||||||
git.defalsify.org/vise.git v0.2.3-0.20250120121301-10739fb4a8c9
|
|
||||||
github.com/jackc/pgx/v5 v5.7.1
|
|
||||||
github.com/joho/godotenv v1.5.1
|
|
||||||
)
|
|
||||||
|
|
||||||
require (
|
|
||||||
github.com/barbashov/iso639-3 v0.0.0-20211020172741-1f4ffb2d8d1c // indirect
|
|
||||||
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect
|
|
||||||
github.com/fxamacker/cbor/v2 v2.4.0 // indirect
|
|
||||||
github.com/graygnuorg/go-gdbm v0.0.0-20220711140707-71387d66dce4 // indirect
|
|
||||||
github.com/jackc/pgpassfile v1.0.0 // indirect
|
|
||||||
github.com/jackc/pgservicefile v0.0.0-20240606120523-5a60cdf6a761 // indirect
|
|
||||||
github.com/jackc/puddle/v2 v2.2.2 // indirect
|
|
||||||
github.com/mattn/kinako v0.0.0-20170717041458-332c0a7e205a // indirect
|
|
||||||
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect
|
|
||||||
github.com/stretchr/testify v1.9.0 // indirect
|
|
||||||
github.com/x448/float16 v0.8.4 // indirect
|
|
||||||
golang.org/x/crypto v0.27.0 // indirect
|
|
||||||
golang.org/x/sync v0.8.0 // indirect
|
|
||||||
golang.org/x/text v0.18.0 // indirect
|
|
||||||
gopkg.in/leonelquinteros/gotext.v1 v1.3.1 // indirect
|
|
||||||
)
|
|
||||||
|
47
go.sum
47
go.sum
@ -1,49 +1,2 @@
|
|||||||
git.defalsify.org/vise.git v0.2.3-0.20250120121301-10739fb4a8c9 h1:sPcqXQcywxA8W3W+9qQncLPmsrgqTIlec7vmD4/7vyA=
|
|
||||||
git.defalsify.org/vise.git v0.2.3-0.20250120121301-10739fb4a8c9/go.mod h1:jyBMe1qTYUz3mmuoC9JQ/TvFeW0vTanCUcPu3H8p4Ck=
|
|
||||||
github.com/barbashov/iso639-3 v0.0.0-20211020172741-1f4ffb2d8d1c h1:H9Nm+I7Cg/YVPpEV1RzU3Wq2pjamPc/UtHDgItcb7lE=
|
|
||||||
github.com/barbashov/iso639-3 v0.0.0-20211020172741-1f4ffb2d8d1c/go.mod h1:rGod7o6KPeJ+hyBpHfhi4v7blx9sf+QsHsA7KAsdN6U=
|
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
|
||||||
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM=
|
|
||||||
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
|
||||||
github.com/fxamacker/cbor/v2 v2.4.0 h1:ri0ArlOR+5XunOP8CRUowT0pSJOwhW098ZCUyskZD88=
|
|
||||||
github.com/fxamacker/cbor/v2 v2.4.0/go.mod h1:TA1xS00nchWmaBnEIxPSE5oHLuJBAVvqrtAnWBwBCVo=
|
|
||||||
github.com/graygnuorg/go-gdbm v0.0.0-20220711140707-71387d66dce4 h1:U4kkNYryi/qfbBF8gh7Vsbuz+cVmhf5kt6pE9bYYyLo=
|
|
||||||
github.com/graygnuorg/go-gdbm v0.0.0-20220711140707-71387d66dce4/go.mod h1:zpZDgZFzeq9s0MIeB1P50NIEWDFFHSFBohI/NbaTD/Y=
|
|
||||||
github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM=
|
|
||||||
github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg=
|
|
||||||
github.com/jackc/pgservicefile v0.0.0-20240606120523-5a60cdf6a761 h1:iCEnooe7UlwOQYpKFhBabPMi4aNAfoODPEFNiAnClxo=
|
|
||||||
github.com/jackc/pgservicefile v0.0.0-20240606120523-5a60cdf6a761/go.mod h1:5TJZWKEWniPve33vlWYSoGYefn3gLQRzjfDlhSJ9ZKM=
|
|
||||||
github.com/jackc/pgx/v5 v5.7.1 h1:x7SYsPBYDkHDksogeSmZZ5xzThcTgRz++I5E+ePFUcs=
|
|
||||||
github.com/jackc/pgx/v5 v5.7.1/go.mod h1:e7O26IywZZ+naJtWWos6i6fvWK+29etgITqrqHLfoZA=
|
|
||||||
github.com/jackc/puddle/v2 v2.2.2 h1:PR8nw+E/1w0GLuRFSmiioY6UooMp6KJv0/61nB7icHo=
|
|
||||||
github.com/jackc/puddle/v2 v2.2.2/go.mod h1:vriiEXHvEE654aYKXXjOvZM39qJ0q+azkZFrfEOc3H4=
|
|
||||||
github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0=
|
|
||||||
github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4=
|
|
||||||
github.com/mattn/kinako v0.0.0-20170717041458-332c0a7e205a h1:0Q3H0YXzMHiciXtRcM+j0jiCe8WKPQHoRgQiRTnfcLY=
|
|
||||||
github.com/mattn/kinako v0.0.0-20170717041458-332c0a7e205a/go.mod h1:CdTTBOYzS5E4mWS1N8NWP6AHI19MP0A2B18n3hLzRMk=
|
|
||||||
github.com/pashagolub/pgxmock/v4 v4.3.0 h1:DqT7fk0OCK6H0GvqtcMsLpv8cIwWqdxWgfZNLeHCb/s=
|
|
||||||
github.com/pashagolub/pgxmock/v4 v4.3.0/go.mod h1:9VoVHXwS3XR/yPtKGzwQvwZX1kzGB9sM8SviDcHDa3A=
|
|
||||||
github.com/peteole/testdata-loader v0.3.0 h1:8jckE9KcyNHgyv/VPoaljvKZE0Rqr8+dPVYH6rfNr9I=
|
|
||||||
github.com/peteole/testdata-loader v0.3.0/go.mod h1:Mt0ZbRtb56u8SLJpNP+BnQbENljMorYBpqlvt3cS83U=
|
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
|
||||||
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRIccs7FGNTlIRMkT8wgtp5eCXdBlqhYGL6U=
|
|
||||||
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
|
||||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
|
||||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
|
||||||
github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
|
|
||||||
github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg=
|
github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg=
|
||||||
github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
|
github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
|
||||||
github.com/x448/float16 v0.8.4 h1:qLwI1I70+NjRFUR3zs1JPUCgaCXSh3SW62uAKT1mSBM=
|
|
||||||
github.com/x448/float16 v0.8.4/go.mod h1:14CWIYCyZA/cWjXOioeEpHeN/83MdbZDRQHoFcYsOfg=
|
|
||||||
golang.org/x/crypto v0.27.0 h1:GXm2NjJrPaiv/h1tb2UH8QfgC/hOf/+z0p6PT8o1w7A=
|
|
||||||
golang.org/x/crypto v0.27.0/go.mod h1:1Xngt8kV6Dvbssa53Ziq6Eqn0HqbZi5Z6R0ZpwQzt70=
|
|
||||||
golang.org/x/sync v0.8.0 h1:3NFvSEYkUoMifnESzZl15y791HH1qU2xm6eCJU5ZPXQ=
|
|
||||||
golang.org/x/sync v0.8.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
|
|
||||||
golang.org/x/text v0.18.0 h1:XvMDiNzPAl0jr17s6W9lcaIhGUfUORdGCNsuLmPG224=
|
|
||||||
golang.org/x/text v0.18.0/go.mod h1:BuEKDfySbSR4drPmRPG/7iBdf8hvFMuRexcpahXilzY=
|
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
|
||||||
gopkg.in/leonelquinteros/gotext.v1 v1.3.1 h1:8d9/fdTG0kn/B7NNGV1BsEyvektXFAbkMsTZS2sFSCc=
|
|
||||||
gopkg.in/leonelquinteros/gotext.v1 v1.3.1/go.mod h1:X1WlGDeAFIYsW6GjgMm4VwUwZ2XjI7Zan2InxSUQWrU=
|
|
||||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
|
||||||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
|
||||||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
|
||||||
|
112
internal/handlers/server/accountservice.go
Normal file
112
internal/handlers/server/accountservice.go
Normal file
@ -0,0 +1,112 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
"git.grassecon.net/urdt/ussd/config"
|
||||||
|
"git.grassecon.net/urdt/ussd/internal/models"
|
||||||
|
)
|
||||||
|
|
||||||
|
type AccountServiceInterface interface {
|
||||||
|
CheckBalance(publicKey string) (string, error)
|
||||||
|
CreateAccount() (*models.AccountResponse, error)
|
||||||
|
CheckAccountStatus(trackingId string) (string, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type AccountService struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// CheckAccountStatus retrieves the status of an account transaction based on the provided tracking ID.
|
||||||
|
//
|
||||||
|
// Parameters:
|
||||||
|
// - trackingId: A unique identifier for the account.This should be obtained from a previous call to
|
||||||
|
// CreateAccount or a similar function that returns an AccountResponse. The `trackingId` field in the
|
||||||
|
// AccountResponse struct can be used here to check the account status during a transaction.
|
||||||
|
//
|
||||||
|
//
|
||||||
|
// Returns:
|
||||||
|
// - string: The status of the transaction as a string. If there is an error during the request or processing, this will be an empty string.
|
||||||
|
// - error: An error if any occurred during the HTTP request, reading the response, or unmarshalling the JSON data.
|
||||||
|
// If no error occurs, this will be nil.
|
||||||
|
//
|
||||||
|
func (as *AccountService) CheckAccountStatus(trackingId string) (string, error) {
|
||||||
|
resp, err := http.Get(config.TrackStatusURL + trackingId)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
body, err := io.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
var trackResp models.TrackStatusResponse
|
||||||
|
err = json.Unmarshal(body, &trackResp)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
status := trackResp.Result.Transaction.Status
|
||||||
|
|
||||||
|
return status, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// CheckBalance retrieves the balance for a given public key from the custodial balance API endpoint.
|
||||||
|
// Parameters:
|
||||||
|
// - publicKey: The public key associated with the account whose balance needs to be checked.
|
||||||
|
func (as *AccountService) CheckBalance(publicKey string) (string, error) {
|
||||||
|
|
||||||
|
resp, err := http.Get(config.BalanceURL + publicKey)
|
||||||
|
if err != nil {
|
||||||
|
return "0.0", err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
body, err := io.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return "0.0", err
|
||||||
|
}
|
||||||
|
|
||||||
|
var balanceResp models.BalanceResponse
|
||||||
|
err = json.Unmarshal(body, &balanceResp)
|
||||||
|
if err != nil {
|
||||||
|
return "0.0", err
|
||||||
|
}
|
||||||
|
|
||||||
|
balance := balanceResp.Result.Balance
|
||||||
|
return balance, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//CreateAccount creates a new account in the custodial system.
|
||||||
|
// Returns:
|
||||||
|
// - *models.AccountResponse: A pointer to an AccountResponse struct containing the details of the created account.
|
||||||
|
// If there is an error during the request or processing, this will be nil.
|
||||||
|
// - error: An error if any occurred during the HTTP request, reading the response, or unmarshalling the JSON data.
|
||||||
|
// If no error occurs, this will be nil.
|
||||||
|
func (as *AccountService) CreateAccount() (*models.AccountResponse, error) {
|
||||||
|
resp, err := http.Post(config.CreateAccountURL, "application/json", nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
body, err := io.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var accountResp models.AccountResponse
|
||||||
|
err = json.Unmarshal(body, &accountResp)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return &accountResp, nil
|
||||||
|
}
|
909
internal/handlers/ussd/menuhandler.go
Normal file
909
internal/handlers/ussd/menuhandler.go
Normal file
@ -0,0 +1,909 @@
|
|||||||
|
package ussd
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"context"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"path"
|
||||||
|
"regexp"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"git.defalsify.org/vise.git/asm"
|
||||||
|
"git.defalsify.org/vise.git/engine"
|
||||||
|
"git.defalsify.org/vise.git/lang"
|
||||||
|
"git.defalsify.org/vise.git/resource"
|
||||||
|
"git.defalsify.org/vise.git/state"
|
||||||
|
"git.grassecon.net/urdt/ussd/internal/handlers/server"
|
||||||
|
"git.grassecon.net/urdt/ussd/internal/utils"
|
||||||
|
"github.com/graygnuorg/go-gdbm"
|
||||||
|
"gopkg.in/leonelquinteros/gotext.v1"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
scriptDir = path.Join("services", "registration")
|
||||||
|
translationDir = path.Join(scriptDir, "locale")
|
||||||
|
//dbFile = path.Join(scriptDir, "userdata.gdbm")
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
TrackingIdKey = "TRACKINGID"
|
||||||
|
PublicKeyKey = "PUBLICKEY"
|
||||||
|
CustodialIdKey = "CUSTODIALID"
|
||||||
|
AccountPin = "ACCOUNTPIN"
|
||||||
|
AccountStatus = "ACCOUNTSTATUS"
|
||||||
|
FirstName = "FIRSTNAME"
|
||||||
|
FamilyName = "FAMILYNAME"
|
||||||
|
YearOfBirth = "YOB"
|
||||||
|
Location = "LOCATION"
|
||||||
|
Gender = "GENDER"
|
||||||
|
Offerings = "OFFERINGS"
|
||||||
|
Recipient = "RECIPIENT"
|
||||||
|
Amount = "AMOUNT"
|
||||||
|
AccountCreated = "ACCOUNTCREATED"
|
||||||
|
)
|
||||||
|
|
||||||
|
func toBytes(s string) []byte {
|
||||||
|
return []byte(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
type FSData struct {
|
||||||
|
Path string
|
||||||
|
St *state.State
|
||||||
|
}
|
||||||
|
|
||||||
|
type FlagParserInterface interface {
|
||||||
|
GetFlag(key string) (uint32, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type Handlers struct {
|
||||||
|
fs *FSData
|
||||||
|
db *gdbm.Database
|
||||||
|
parser FlagParserInterface
|
||||||
|
accountFileHandler utils.AccountFileHandlerInterface
|
||||||
|
accountService server.AccountServiceInterface
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewHandlers(dir string, st *state.State, sessionId string) (*Handlers, error) {
|
||||||
|
filename := path.Join(scriptDir, sessionId+"_userdata.gdbm")
|
||||||
|
db, err := gdbm.Open(filename, gdbm.ModeWrcreat)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
pfp := path.Join(scriptDir, "pp.csv")
|
||||||
|
parser := asm.NewFlagParser()
|
||||||
|
_, err = parser.Load(pfp)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &Handlers{
|
||||||
|
db: db,
|
||||||
|
fs: &FSData{
|
||||||
|
Path: dir,
|
||||||
|
St: st,
|
||||||
|
},
|
||||||
|
parser: parser,
|
||||||
|
accountFileHandler: utils.NewAccountFileHandler(dir + "_data"),
|
||||||
|
accountService: &server.AccountService{},
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Define the regex pattern as a constant
|
||||||
|
const pinPattern = `^\d{4}$`
|
||||||
|
|
||||||
|
// isValidPIN checks whether the given input is a 4 digit number
|
||||||
|
func isValidPIN(pin string) bool {
|
||||||
|
match, _ := regexp.MatchString(pinPattern, pin)
|
||||||
|
return match
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *Handlers) PreloadFlags(flagKeys []string) (map[string]uint32, error) {
|
||||||
|
flags := make(map[string]uint32)
|
||||||
|
for _, key := range flagKeys {
|
||||||
|
flag, err := h.parser.GetFlag(key)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
flags[key] = flag
|
||||||
|
}
|
||||||
|
return flags, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetLanguage sets the language across the menu
|
||||||
|
func (h *Handlers) SetLanguage(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
// Preload the required flag
|
||||||
|
flagKeys := []string{"flag_language_set"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
inputStr := string(input)
|
||||||
|
switch inputStr {
|
||||||
|
case "0":
|
||||||
|
res.FlagSet = []uint32{state.FLAG_LANG}
|
||||||
|
res.Content = "eng"
|
||||||
|
case "1":
|
||||||
|
res.FlagSet = []uint32{state.FLAG_LANG}
|
||||||
|
res.Content = "swa"
|
||||||
|
default:
|
||||||
|
}
|
||||||
|
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_language_set"])
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// CreateAccount checks if any account exists on the JSON data file, and if not
|
||||||
|
// creates an account on the API,
|
||||||
|
// sets the default values and flags
|
||||||
|
func (h *Handlers) CreateAccount(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
// Preload the required flags
|
||||||
|
flagKeys := []string{"flag_account_created", "flag_account_creation_failed"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
_, err = h.db.Fetch([]byte(AccountCreated))
|
||||||
|
if err != nil {
|
||||||
|
if errors.Is(err, gdbm.ErrItemNotFound) {
|
||||||
|
accountResp, err := h.accountService.CreateAccount()
|
||||||
|
if err != nil {
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_account_creation_failed"])
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
data := map[string]string{
|
||||||
|
TrackingIdKey: accountResp.Result.TrackingId,
|
||||||
|
PublicKeyKey: accountResp.Result.PublicKey,
|
||||||
|
CustodialIdKey: accountResp.Result.CustodialId.String(),
|
||||||
|
}
|
||||||
|
|
||||||
|
for key, value := range data {
|
||||||
|
err := h.db.Store(toBytes(key), toBytes(value), true)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
key := []byte(AccountCreated)
|
||||||
|
value := []byte("1")
|
||||||
|
h.db.Store(key, value, true)
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_account_created"])
|
||||||
|
return res, err
|
||||||
|
} else {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// SavePin persists the user's PIN choice into the filesystem
|
||||||
|
func (h *Handlers) SavePin(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
flagKeys := []string{"flag_incorrect_pin"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
accountPIN := string(input)
|
||||||
|
// Validate that the PIN is a 4-digit number
|
||||||
|
if !isValidPIN(accountPIN) {
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_incorrect_pin"])
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_incorrect_pin"])
|
||||||
|
|
||||||
|
key := []byte(AccountPin)
|
||||||
|
value := []byte(accountPIN)
|
||||||
|
|
||||||
|
h.db.Store(key, value, true)
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SetResetSingleEdit sets and resets flags to allow gradual editing of profile information.
|
||||||
|
func (h *Handlers) SetResetSingleEdit(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
menuOption := string(input)
|
||||||
|
|
||||||
|
// Preload the required flags
|
||||||
|
flagKeys := []string{"flag_allow_update", "flag_single_edit"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
switch menuOption {
|
||||||
|
case "2":
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_allow_update"])
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_single_edit"])
|
||||||
|
case "3":
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_allow_update"])
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_single_edit"])
|
||||||
|
case "4":
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_allow_update"])
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_single_edit"])
|
||||||
|
default:
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_single_edit"])
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// VerifyPin checks whether the confirmation PIN is similar to the account PIN
|
||||||
|
// If similar, it sets the USERFLAG_PIN_SET flag allowing the user
|
||||||
|
// to access the main menu
|
||||||
|
func (h *Handlers) VerifyPin(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
// Preload the required flags
|
||||||
|
flagKeys := []string{"flag_valid_pin", "flag_pin_mismatch", "flag_pin_set"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
AccountPin, err := h.db.Fetch([]byte(AccountPin))
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
if bytes.Equal(input, AccountPin) {
|
||||||
|
res.FlagSet = []uint32{flags["flag_valid_pin"]}
|
||||||
|
res.FlagReset = []uint32{flags["flag_pin_mismatch"]}
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_pin_set"])
|
||||||
|
} else {
|
||||||
|
res.FlagSet = []uint32{flags["flag_pin_mismatch"]}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// codeFromCtx retrieves language codes from the context that can be used for handling translations
|
||||||
|
func codeFromCtx(ctx context.Context) string {
|
||||||
|
var code string
|
||||||
|
engine.Logg.DebugCtxf(ctx, "in msg", "ctx", ctx, "val", code)
|
||||||
|
if ctx.Value("Language") != nil {
|
||||||
|
lang := ctx.Value("Language").(lang.Language)
|
||||||
|
code = lang.Code
|
||||||
|
}
|
||||||
|
return code
|
||||||
|
}
|
||||||
|
|
||||||
|
// SaveFirstname updates the first name in a JSON data file with the provided input.
|
||||||
|
func (h *Handlers) SaveFirstname(cxt context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
if len(input) > 0 {
|
||||||
|
name := string(input)
|
||||||
|
key := []byte(FirstName)
|
||||||
|
value := []byte(name)
|
||||||
|
h.db.Store(key, value, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SaveFamilyname updates the family name in a JSON data file with the provided input.
|
||||||
|
func (h *Handlers) SaveFamilyname(cxt context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
if len(input) > 0 {
|
||||||
|
secondname := string(input)
|
||||||
|
key := []byte(FamilyName)
|
||||||
|
value := []byte(secondname)
|
||||||
|
h.db.Store(key, value, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SaveYOB updates the Year of Birth(YOB) in a JSON data file with the provided input.
|
||||||
|
func (h *Handlers) SaveYob(cxt context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
yob := string(input)
|
||||||
|
if len(yob) == 4 {
|
||||||
|
yob := string(input)
|
||||||
|
key := []byte(YearOfBirth)
|
||||||
|
value := []byte(yob)
|
||||||
|
h.db.Store(key, value, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SaveLocation updates the location in a JSON data file with the provided input.
|
||||||
|
func (h *Handlers) SaveLocation(cxt context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
if len(input) > 0 {
|
||||||
|
location := string(input)
|
||||||
|
key := []byte(Location)
|
||||||
|
value := []byte(location)
|
||||||
|
|
||||||
|
h.db.Store(key, value, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SaveGender updates the gender in a JSON data file with the provided input.
|
||||||
|
func (h *Handlers) SaveGender(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
if len(input) > 0 {
|
||||||
|
gender := string(input)
|
||||||
|
switch gender {
|
||||||
|
case "1":
|
||||||
|
gender = "Male"
|
||||||
|
case "2":
|
||||||
|
gender = "Female"
|
||||||
|
case "3":
|
||||||
|
gender = "Unspecified"
|
||||||
|
}
|
||||||
|
key := []byte(Gender)
|
||||||
|
value := []byte(gender)
|
||||||
|
h.db.Store(key, value, true)
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// SaveOfferings updates the offerings(goods and services provided by the user) in a JSON data file with the provided input.
|
||||||
|
func (h *Handlers) SaveOfferings(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
if len(input) > 0 {
|
||||||
|
offerings := string(input)
|
||||||
|
key := []byte(Offerings)
|
||||||
|
value := []byte(offerings)
|
||||||
|
h.db.Store(key, value, true)
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ResetAllowUpdate resets the allowupdate flag that allows a user to update profile data.
|
||||||
|
func (h *Handlers) ResetAllowUpdate(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
// Preload the required flag
|
||||||
|
flagKeys := []string{"flag_allow_update"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_allow_update"])
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ResetAccountAuthorized resets the account authorization flag after a successful PIN entry.
|
||||||
|
func (h *Handlers) ResetAccountAuthorized(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
// Preload the required flags
|
||||||
|
flagKeys := []string{"flag_account_authorized"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_account_authorized"])
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// CheckIdentifier retrieves the PublicKey from the JSON data file.
|
||||||
|
func (h *Handlers) CheckIdentifier(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
publicKey, err := h.db.Fetch([]byte(PublicKeyKey))
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
res.Content = string(publicKey)
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Authorize attempts to unlock the next sequential nodes by verifying the provided PIN against the already set PIN.
|
||||||
|
// It sets the required flags that control the flow.
|
||||||
|
func (h *Handlers) Authorize(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
// Preload the required flags
|
||||||
|
flagKeys := []string{"flag_incorrect_pin", "flag_account_authorized", "flag_allow_update"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
storedpin, err := h.db.Fetch([]byte(AccountPin))
|
||||||
|
if err == nil {
|
||||||
|
if len(input) == 4 {
|
||||||
|
if bytes.Equal(input, storedpin) {
|
||||||
|
if h.fs.St.MatchFlag(flags["flag_account_authorized"], false) {
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_incorrect_pin"])
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_allow_update"], flags["flag_account_authorized"])
|
||||||
|
} else {
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_allow_update"])
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_account_authorized"])
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_incorrect_pin"])
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_account_authorized"])
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if errors.Is(err, gdbm.ErrItemNotFound) {
|
||||||
|
return res, err
|
||||||
|
} else {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
// Preload the required flag
|
||||||
|
flagKeys := []string{"flag_incorrect_pin"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_incorrect_pin"])
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// CheckAccountStatus queries the API using the TrackingId and sets flags
|
||||||
|
// based on the account status
|
||||||
|
func (h *Handlers) CheckAccountStatus(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
// Preload the required flags
|
||||||
|
flagKeys := []string{"flag_account_success", "flag_account_pending"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
trackingId, err := h.db.Fetch([]byte(TrackingIdKey))
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
status, err := h.accountService.CheckAccountStatus(string(trackingId))
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("Error checking account status:", err)
|
||||||
|
return res, err
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
err = h.db.Store(toBytes(AccountStatus), toBytes(status), true)
|
||||||
|
if err != nil {
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
err = h.db.Store(toBytes(TrackingIdKey), toBytes(status), true)
|
||||||
|
if err != nil {
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if status == "SUCCESS" {
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_account_success"])
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_account_pending"])
|
||||||
|
} else {
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_account_success"])
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_account_pending"])
|
||||||
|
}
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Quit displays the Thank you message and exits the menu
|
||||||
|
func (h *Handlers) Quit(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
// Preload the required flags
|
||||||
|
flagKeys := []string{"flag_account_authorized"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
code := codeFromCtx(ctx)
|
||||||
|
l := gotext.NewLocale(translationDir, code)
|
||||||
|
l.AddDomain("default")
|
||||||
|
|
||||||
|
res.Content = l.Get("Thank you for using Sarafu. Goodbye!")
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_account_authorized"])
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// VerifyYob verifies the length of the given input
|
||||||
|
func (h *Handlers) VerifyYob(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
// Preload the required flag
|
||||||
|
flagKeys := []string{"flag_incorrect_date_format"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
date := string(input)
|
||||||
|
_, err = strconv.Atoi(date)
|
||||||
|
if err != nil {
|
||||||
|
// If conversion fails, input is not numeric
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_incorrect_date_format"])
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(date) == 4 {
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_incorrect_date_format"])
|
||||||
|
} else {
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_incorrect_date_format"])
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ResetIncorrectYob resets the incorrect date format flag after a new attempt
|
||||||
|
func (h *Handlers) ResetIncorrectYob(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
// Preload the required flags
|
||||||
|
flagKeys := []string{"flag_incorrect_date_format"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_incorrect_date_format"])
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// CheckBalance retrieves the balance from the API using the "PublicKey" and sets
|
||||||
|
// the balance as the result content
|
||||||
|
func (h *Handlers) CheckBalance(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
publicKey, err := h.db.Fetch([]byte(PublicKeyKey))
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
balance, err := h.accountService.CheckBalance(string(publicKey))
|
||||||
|
if err != nil {
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
res.Content = balance
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ValidateRecipient validates that the given input is a valid phone number.
|
||||||
|
func (h *Handlers) ValidateRecipient(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
recipient := string(input)
|
||||||
|
|
||||||
|
flagKeys := []string{"flag_invalid_recipient"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if recipient != "0" {
|
||||||
|
// mimic invalid number check
|
||||||
|
if recipient == "000" {
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_invalid_recipient"])
|
||||||
|
res.Content = recipient
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// accountData["Recipient"] = recipient
|
||||||
|
key := []byte(Recipient)
|
||||||
|
value := []byte(recipient)
|
||||||
|
|
||||||
|
h.db.Store(key, value, true)
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// TransactionReset resets the previous transaction data (Recipient and Amount)
|
||||||
|
// as well as the invalid flags
|
||||||
|
func (h *Handlers) TransactionReset(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
// Preload the required flags
|
||||||
|
flagKeys := []string{"flag_invalid_recipient", "flag_invalid_recipient_with_invite"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = h.db.Delete([]byte(Amount))
|
||||||
|
if err != nil && !errors.Is(err, gdbm.ErrItemNotFound) {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
err = h.db.Delete([]byte(Recipient))
|
||||||
|
if err != nil && !errors.Is(err, gdbm.ErrItemNotFound) {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_invalid_recipient"], flags["flag_invalid_recipient_with_invite"])
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ResetTransactionAmount resets the transaction amount and invalid flag
|
||||||
|
func (h *Handlers) ResetTransactionAmount(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
// Preload the required flag
|
||||||
|
flagKeys := []string{"flag_invalid_amount"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
err = h.db.Delete([]byte(Amount))
|
||||||
|
if err != nil && !errors.Is(err, gdbm.ErrItemNotFound) {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_invalid_amount"])
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// MaxAmount gets the current balance from the API and sets it as
|
||||||
|
// the result content.
|
||||||
|
func (h *Handlers) MaxAmount(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
publicKey, err := h.db.Fetch([]byte(PublicKeyKey))
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
balance, err := h.accountService.CheckBalance(string(publicKey))
|
||||||
|
if err != nil {
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
res.Content = balance
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ValidateAmount ensures that the given input is a valid amount and that
|
||||||
|
// it is not more than the current balance.
|
||||||
|
func (h *Handlers) ValidateAmount(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
// Preload the required flag
|
||||||
|
flagKeys := []string{"flag_invalid_amount"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
amountStr := string(input)
|
||||||
|
publicKey, err := h.db.Fetch([]byte(PublicKeyKey))
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
balanceStr, err := h.accountService.CheckBalance(string(publicKey))
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
res.Content = balanceStr
|
||||||
|
|
||||||
|
// Parse the balance
|
||||||
|
balanceParts := strings.Split(balanceStr, " ")
|
||||||
|
if len(balanceParts) != 2 {
|
||||||
|
return res, fmt.Errorf("unexpected balance format: %s", balanceStr)
|
||||||
|
}
|
||||||
|
balanceValue, err := strconv.ParseFloat(balanceParts[0], 64)
|
||||||
|
if err != nil {
|
||||||
|
return res, fmt.Errorf("failed to parse balance: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Extract numeric part from input
|
||||||
|
re := regexp.MustCompile(`^(\d+(\.\d+)?)\s*(?:CELO)?$`)
|
||||||
|
matches := re.FindStringSubmatch(strings.TrimSpace(amountStr))
|
||||||
|
if len(matches) < 2 {
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_invalid_amount"])
|
||||||
|
res.Content = amountStr
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
inputAmount, err := strconv.ParseFloat(matches[1], 64)
|
||||||
|
if err != nil {
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_invalid_amount"])
|
||||||
|
res.Content = amountStr
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if inputAmount > balanceValue {
|
||||||
|
res.FlagSet = append(res.FlagSet, flags["flag_invalid_amount"])
|
||||||
|
res.Content = amountStr
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
res.Content = fmt.Sprintf("%.3f", inputAmount) // Format to 3 decimal places
|
||||||
|
key := []byte(Amount)
|
||||||
|
value := []byte(res.Content)
|
||||||
|
h.db.Store(key, value, true)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetRecipient returns the transaction recipient from a JSON data file.
|
||||||
|
func (h *Handlers) GetRecipient(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
recipient, err := h.db.Fetch([]byte(Recipient))
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
res.Content = string(recipient)
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetSender retrieves the public key from a JSON data file.
|
||||||
|
func (h *Handlers) GetSender(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
//accountData, err := h.accountFileHandler.ReadAccountData()
|
||||||
|
publicKey, err := h.db.Fetch([]byte(PublicKeyKey))
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
res.Content = string(publicKey)
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetAmount retrieves the amount from a JSON data file.
|
||||||
|
func (h *Handlers) GetAmount(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
amount, err := h.db.Fetch([]byte(Amount))
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
res.Content = string(amount)
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// QuickWithBalance retrieves the balance for a given public key from the custodial balance API endpoint before
|
||||||
|
// gracefully exiting the session.
|
||||||
|
func (h *Handlers) QuitWithBalance(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
// Preload the required flag
|
||||||
|
flagKeys := []string{"flag_account_authorized"}
|
||||||
|
flags, err := h.PreloadFlags(flagKeys)
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
code := codeFromCtx(ctx)
|
||||||
|
l := gotext.NewLocale(translationDir, code)
|
||||||
|
l.AddDomain("default")
|
||||||
|
publicKey, err := h.db.Fetch([]byte(PublicKeyKey))
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
balance, err := h.accountService.CheckBalance(string(publicKey))
|
||||||
|
if err != nil {
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
res.Content = l.Get("Your account balance is %s", balance)
|
||||||
|
res.FlagReset = append(res.FlagReset, flags["flag_account_authorized"])
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// InitiateTransaction returns a confirmation and resets the transaction data
|
||||||
|
// on the JSON file.
|
||||||
|
func (h *Handlers) InitiateTransaction(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
code := codeFromCtx(ctx)
|
||||||
|
l := gotext.NewLocale(translationDir, code)
|
||||||
|
l.AddDomain("default")
|
||||||
|
// TODO
|
||||||
|
// Use the amount, recipient and sender to call the API and initialize the transaction
|
||||||
|
|
||||||
|
publicKey, err := h.db.Fetch([]byte(PublicKeyKey))
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
amount, err := h.db.Fetch([]byte(Amount))
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
recipient, err := h.db.Fetch([]byte(Recipient))
|
||||||
|
if err != nil {
|
||||||
|
return res, err
|
||||||
|
}
|
||||||
|
|
||||||
|
res.Content = l.Get("Your request has been sent. %s will receive %s from %s.", string(recipient), string(amount), string(publicKey))
|
||||||
|
|
||||||
|
account_authorized_flag, err := h.parser.GetFlag("flag_account_authorized")
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
res.FlagReset = append(res.FlagReset, account_authorized_flag)
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetProfileInfo retrieves and formats the profile information of a user from a Gdbm backed storage.
|
||||||
|
func (h *Handlers) GetProfileInfo(ctx context.Context, sym string, input []byte) (resource.Result, error) {
|
||||||
|
res := resource.Result{}
|
||||||
|
|
||||||
|
// Define default values
|
||||||
|
defaultValue := "Not provided"
|
||||||
|
name := defaultValue
|
||||||
|
familyName := defaultValue
|
||||||
|
yob := defaultValue
|
||||||
|
gender := defaultValue
|
||||||
|
location := defaultValue
|
||||||
|
offerings := defaultValue
|
||||||
|
|
||||||
|
// Fetch data using a map for better organization
|
||||||
|
dataKeys := map[string]*string{
|
||||||
|
FirstName: &name,
|
||||||
|
FamilyName: &familyName,
|
||||||
|
YearOfBirth: &yob,
|
||||||
|
Location: &location,
|
||||||
|
Gender: &gender,
|
||||||
|
Offerings: &offerings,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Iterate over keys and fetch values
|
||||||
|
//iter := h.db.Iterator()
|
||||||
|
next := h.db.Iterator()
|
||||||
|
//defer iter.Close() // Ensure the iterator is closed
|
||||||
|
for key, err := next(); err == nil; key, err = next() {
|
||||||
|
if valuePointer, ok := dataKeys[string(key)]; ok {
|
||||||
|
value, fetchErr := h.db.Fetch(key)
|
||||||
|
if fetchErr == nil {
|
||||||
|
*valuePointer = string(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Construct the full name
|
||||||
|
if familyName != defaultValue {
|
||||||
|
if name == defaultValue {
|
||||||
|
name = familyName
|
||||||
|
} else {
|
||||||
|
name = name + " " + familyName
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculate age from year of birth
|
||||||
|
var age string
|
||||||
|
if yob != defaultValue {
|
||||||
|
yobInt, err := strconv.Atoi(yob)
|
||||||
|
if err != nil {
|
||||||
|
return res, fmt.Errorf("invalid year of birth: %v", err)
|
||||||
|
}
|
||||||
|
age = strconv.Itoa(utils.CalculateAgeWithYOB(yobInt))
|
||||||
|
} else {
|
||||||
|
age = defaultValue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format the result
|
||||||
|
formattedData := fmt.Sprintf("Name: %s\nGender: %s\nAge: %s\nLocation: %s\nYou provide: %s\n", name, gender, age, location, offerings)
|
||||||
|
res.Content = formattedData
|
||||||
|
return res, nil
|
||||||
|
}
|
993
internal/handlers/ussd/menuhandler_test.go
Normal file
993
internal/handlers/ussd/menuhandler_test.go
Normal file
@ -0,0 +1,993 @@
|
|||||||
|
package ussd
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"git.defalsify.org/vise.git/resource"
|
||||||
|
"git.grassecon.net/urdt/ussd/internal/handlers/ussd/mocks"
|
||||||
|
"git.grassecon.net/urdt/ussd/internal/models"
|
||||||
|
"git.grassecon.net/urdt/ussd/internal/utils"
|
||||||
|
"github.com/alecthomas/assert/v2"
|
||||||
|
"github.com/stretchr/testify/mock"
|
||||||
|
)
|
||||||
|
|
||||||
|
// MockAccountService implements AccountServiceInterface for testing
|
||||||
|
type MockAccountService struct {
|
||||||
|
mock.Mock
|
||||||
|
}
|
||||||
|
|
||||||
|
type MockFlagParser struct {
|
||||||
|
mock.Mock
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockFlagParser) GetFlag(key string) (uint32, error) {
|
||||||
|
args := m.Called(key)
|
||||||
|
return args.Get(0).(uint32), args.Error(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockAccountService) CreateAccount() (*models.AccountResponse, error) {
|
||||||
|
args := m.Called()
|
||||||
|
return args.Get(0).(*models.AccountResponse), args.Error(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockAccountService) CheckBalance(publicKey string) (string, error) {
|
||||||
|
args := m.Called(publicKey)
|
||||||
|
return args.String(0), args.Error(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockAccountService) CheckAccountStatus(trackingId string) (string, error) {
|
||||||
|
args := m.Called(trackingId)
|
||||||
|
return args.String(0), args.Error(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCreateAccount(t *testing.T) {
|
||||||
|
// Setup
|
||||||
|
tempDir, err := os.MkdirTemp("", "test_create_account")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to create temp directory: %v", err)
|
||||||
|
}
|
||||||
|
defer os.RemoveAll(tempDir) // Clean up after the test run
|
||||||
|
|
||||||
|
sessionID := "07xxxxxxxx"
|
||||||
|
|
||||||
|
// Set up the data file path using the session ID
|
||||||
|
accountFilePath := filepath.Join(tempDir, sessionID+"_data")
|
||||||
|
|
||||||
|
// Initialize account file handler
|
||||||
|
accountFileHandler := utils.NewAccountFileHandler(accountFilePath)
|
||||||
|
|
||||||
|
// Create a mock account service
|
||||||
|
mockAccountService := &MockAccountService{}
|
||||||
|
mockAccountResponse := &models.AccountResponse{
|
||||||
|
Ok: true,
|
||||||
|
Result: struct {
|
||||||
|
CustodialId json.Number `json:"custodialId"`
|
||||||
|
PublicKey string `json:"publicKey"`
|
||||||
|
TrackingId string `json:"trackingId"`
|
||||||
|
}{
|
||||||
|
CustodialId: "test-custodial-id",
|
||||||
|
PublicKey: "test-public-key",
|
||||||
|
TrackingId: "test-tracking-id",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set up expectations for the mock account service
|
||||||
|
mockAccountService.On("CreateAccount").Return(mockAccountResponse, nil)
|
||||||
|
|
||||||
|
mockParser := new(MockFlagParser)
|
||||||
|
|
||||||
|
flag_account_created := uint32(1)
|
||||||
|
flag_account_creation_failed := uint32(2)
|
||||||
|
|
||||||
|
mockParser.On("GetFlag", "flag_account_created").Return(flag_account_created, nil)
|
||||||
|
mockParser.On("GetFlag", "flag_account_creation_failed").Return(flag_account_creation_failed, nil)
|
||||||
|
|
||||||
|
// Initialize Handlers with mock account service
|
||||||
|
h := &Handlers{
|
||||||
|
fs: &FSData{Path: accountFilePath},
|
||||||
|
accountFileHandler: accountFileHandler,
|
||||||
|
accountService: mockAccountService,
|
||||||
|
parser: mockParser,
|
||||||
|
}
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
existingData map[string]string
|
||||||
|
expectedResult resource.Result
|
||||||
|
expectedData map[string]string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "New account creation",
|
||||||
|
existingData: nil,
|
||||||
|
expectedResult: resource.Result{
|
||||||
|
FlagSet: []uint32{flag_account_created},
|
||||||
|
},
|
||||||
|
expectedData: map[string]string{
|
||||||
|
"TrackingId": "test-tracking-id",
|
||||||
|
"PublicKey": "test-public-key",
|
||||||
|
"CustodialId": "test-custodial-id",
|
||||||
|
"Status": "PENDING",
|
||||||
|
"Gender": "Not provided",
|
||||||
|
"YOB": "Not provided",
|
||||||
|
"Location": "Not provided",
|
||||||
|
"Offerings": "Not provided",
|
||||||
|
"FirstName": "Not provided",
|
||||||
|
"FamilyName": "Not provided",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Existing account",
|
||||||
|
existingData: map[string]string{
|
||||||
|
"TrackingId": "test-tracking-id",
|
||||||
|
"PublicKey": "test-public-key",
|
||||||
|
"CustodialId": "test-custodial-id",
|
||||||
|
"Status": "PENDING",
|
||||||
|
"Gender": "Not provided",
|
||||||
|
"YOB": "Not provided",
|
||||||
|
"Location": "Not provided",
|
||||||
|
"Offerings": "Not provided",
|
||||||
|
"FirstName": "Not provided",
|
||||||
|
"FamilyName": "Not provided",
|
||||||
|
},
|
||||||
|
expectedResult: resource.Result{},
|
||||||
|
expectedData: map[string]string{
|
||||||
|
"TrackingId": "test-tracking-id",
|
||||||
|
"PublicKey": "test-public-key",
|
||||||
|
"CustodialId": "test-custodial-id",
|
||||||
|
"Status": "PENDING",
|
||||||
|
"Gender": "Not provided",
|
||||||
|
"YOB": "Not provided",
|
||||||
|
"Location": "Not provided",
|
||||||
|
"Offerings": "Not provided",
|
||||||
|
"FirstName": "Not provided",
|
||||||
|
"FamilyName": "Not provided",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
// Set up the data file path using the session ID
|
||||||
|
accountFilePath := filepath.Join(tempDir, sessionID+"_data")
|
||||||
|
|
||||||
|
// Setup existing data if any
|
||||||
|
if tt.existingData != nil {
|
||||||
|
data, _ := json.Marshal(tt.existingData)
|
||||||
|
err := os.WriteFile(accountFilePath, data, 0644)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to write existing data: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call the function
|
||||||
|
result, err := h.CreateAccount(context.Background(), "", nil)
|
||||||
|
|
||||||
|
// Check for errors
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateAccount returned an error: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check the result
|
||||||
|
if len(result.FlagSet) != len(tt.expectedResult.FlagSet) {
|
||||||
|
t.Errorf("Expected %d flags, got %d", len(tt.expectedResult.FlagSet), len(result.FlagSet))
|
||||||
|
}
|
||||||
|
for i, flag := range tt.expectedResult.FlagSet {
|
||||||
|
if result.FlagSet[i] != flag {
|
||||||
|
t.Errorf("Expected flag %d, got %d", flag, result.FlagSet[i])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check the stored data
|
||||||
|
data, err := os.ReadFile(accountFilePath)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to read account data file: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var storedData map[string]string
|
||||||
|
err = json.Unmarshal(data, &storedData)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to unmarshal stored data: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for key, expectedValue := range tt.expectedData {
|
||||||
|
if storedValue, ok := storedData[key]; !ok || storedValue != expectedValue {
|
||||||
|
t.Errorf("Expected %s to be %s, got %s", key, expectedValue, storedValue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestCreateAccount_Success(t *testing.T) {
|
||||||
|
mockAccountFileHandler := new(mocks.MockAccountFileHandler)
|
||||||
|
mockCreateAccountService := new(mocks.MockAccountService)
|
||||||
|
|
||||||
|
mockAccountFileHandler.On("EnsureFileExists").Return(nil)
|
||||||
|
|
||||||
|
// Mock that no account data exists
|
||||||
|
mockAccountFileHandler.On("ReadAccountData").Return(nil, nil)
|
||||||
|
|
||||||
|
// Define expected account response after api call
|
||||||
|
expectedAccountResp := &models.AccountResponse{
|
||||||
|
Ok: true,
|
||||||
|
Result: struct {
|
||||||
|
CustodialId json.Number `json:"custodialId"`
|
||||||
|
PublicKey string `json:"publicKey"`
|
||||||
|
TrackingId string `json:"trackingId"`
|
||||||
|
}{
|
||||||
|
CustodialId: "12",
|
||||||
|
PublicKey: "0x8E0XSCSVA",
|
||||||
|
TrackingId: "d95a7e83-196c-4fd0-866fSGAGA",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
mockCreateAccountService.On("CreateAccount").Return(expectedAccountResp, nil)
|
||||||
|
|
||||||
|
// Mock WriteAccountData to not error
|
||||||
|
mockAccountFileHandler.On("WriteAccountData", mock.Anything).Return(nil)
|
||||||
|
|
||||||
|
handlers := &Handlers{
|
||||||
|
accountService: mockCreateAccountService,
|
||||||
|
}
|
||||||
|
|
||||||
|
actualResponse, err := handlers.accountService.CreateAccount()
|
||||||
|
|
||||||
|
// Assert results
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, expectedAccountResp.Ok, true)
|
||||||
|
assert.Equal(t, expectedAccountResp, actualResponse)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSavePin(t *testing.T) {
|
||||||
|
// Setup
|
||||||
|
tempDir, err := os.MkdirTemp("", "test_save_pin")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to create temp directory: %v", err)
|
||||||
|
}
|
||||||
|
defer os.RemoveAll(tempDir)
|
||||||
|
|
||||||
|
sessionID := "07xxxxxxxx"
|
||||||
|
|
||||||
|
// Set up the data file path using the session ID
|
||||||
|
accountFilePath := filepath.Join(tempDir, sessionID+"_data")
|
||||||
|
initialAccountData := map[string]string{
|
||||||
|
"TrackingId": "test-tracking-id",
|
||||||
|
"PublicKey": "test-public-key",
|
||||||
|
}
|
||||||
|
data, _ := json.Marshal(initialAccountData)
|
||||||
|
err = os.WriteFile(accountFilePath, data, 0644)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to write initial account data: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create a new AccountFileHandler and set it in the Handlers struct
|
||||||
|
accountFileHandler := utils.NewAccountFileHandler(accountFilePath)
|
||||||
|
mockParser := new(MockFlagParser)
|
||||||
|
|
||||||
|
h := &Handlers{
|
||||||
|
accountFileHandler: accountFileHandler,
|
||||||
|
parser: mockParser,
|
||||||
|
}
|
||||||
|
|
||||||
|
flag_incorrect_pin := uint32(1)
|
||||||
|
mockParser.On("GetFlag", "flag_incorrect_pin").Return(flag_incorrect_pin, nil)
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
input []byte
|
||||||
|
expectedFlags []uint32
|
||||||
|
expectedData map[string]string
|
||||||
|
expectedErrors bool
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "Valid PIN",
|
||||||
|
input: []byte("1234"),
|
||||||
|
expectedFlags: []uint32{},
|
||||||
|
expectedData: map[string]string{
|
||||||
|
"TrackingId": "test-tracking-id",
|
||||||
|
"PublicKey": "test-public-key",
|
||||||
|
"AccountPIN": "1234",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Invalid PIN - non-numeric",
|
||||||
|
input: []byte("12ab"),
|
||||||
|
expectedFlags: []uint32{flag_incorrect_pin},
|
||||||
|
expectedData: initialAccountData, // No changes expected
|
||||||
|
expectedErrors: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Invalid PIN - less than 4 digits",
|
||||||
|
input: []byte("123"),
|
||||||
|
expectedFlags: []uint32{flag_incorrect_pin},
|
||||||
|
expectedData: initialAccountData, // No changes expected
|
||||||
|
expectedErrors: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Invalid PIN - more than 4 digits",
|
||||||
|
input: []byte("12345"),
|
||||||
|
expectedFlags: []uint32{flag_incorrect_pin},
|
||||||
|
expectedData: initialAccountData, // No changes expected
|
||||||
|
expectedErrors: false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
err := accountFileHandler.EnsureFileExists()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to ensure account file exists: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
result, err := h.SavePin(context.Background(), "", tt.input)
|
||||||
|
if err != nil && !tt.expectedErrors {
|
||||||
|
t.Fatalf("SavePin returned an unexpected error: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(result.FlagSet) != len(tt.expectedFlags) {
|
||||||
|
t.Errorf("Expected %d flags, got %d", len(tt.expectedFlags), len(result.FlagSet))
|
||||||
|
}
|
||||||
|
for i, flag := range tt.expectedFlags {
|
||||||
|
if result.FlagSet[i] != flag {
|
||||||
|
t.Errorf("Expected flag %d, got %d", flag, result.FlagSet[i])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
data, err := os.ReadFile(accountFilePath)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to read account data file: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var storedData map[string]string
|
||||||
|
err = json.Unmarshal(data, &storedData)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to unmarshal stored data: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for key, expectedValue := range tt.expectedData {
|
||||||
|
if storedValue, ok := storedData[key]; !ok || storedValue != expectedValue {
|
||||||
|
t.Errorf("Expected %s to be %s, got %s", key, expectedValue, storedValue)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSaveLocation(t *testing.T) {
|
||||||
|
// Create a new instance of MockAccountFileHandler
|
||||||
|
mockFileHandler := new(mocks.MockAccountFileHandler)
|
||||||
|
|
||||||
|
// Define test cases
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
input []byte
|
||||||
|
existingData map[string]string
|
||||||
|
writeError error
|
||||||
|
expectedResult resource.Result
|
||||||
|
expectedError error
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "Successful Save",
|
||||||
|
input: []byte("Mombasa"),
|
||||||
|
existingData: map[string]string{"Location": "Mombasa"},
|
||||||
|
writeError: nil,
|
||||||
|
expectedResult: resource.Result{},
|
||||||
|
expectedError: nil,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Empty location input",
|
||||||
|
input: []byte{},
|
||||||
|
existingData: map[string]string{"OtherKey": "OtherValue"},
|
||||||
|
writeError: nil,
|
||||||
|
expectedResult: resource.Result{},
|
||||||
|
expectedError: nil,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
// Set up the mock expectations
|
||||||
|
mockFileHandler.On("ReadAccountData").Return(tt.existingData, tt.expectedError)
|
||||||
|
if tt.expectedError == nil && len(tt.input) > 0 {
|
||||||
|
mockFileHandler.On("WriteAccountData", mock.MatchedBy(func(data map[string]string) bool {
|
||||||
|
return data["Location"] == string(tt.input)
|
||||||
|
})).Return(tt.writeError)
|
||||||
|
} else if len(tt.input) == 0 {
|
||||||
|
// For empty input, no WriteAccountData call should be made
|
||||||
|
mockFileHandler.On("WriteAccountData", mock.Anything).Maybe().Return(tt.writeError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create the Handlers instance with the mock file handler
|
||||||
|
h := &Handlers{
|
||||||
|
accountFileHandler: mockFileHandler,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call Save Location
|
||||||
|
result, err := h.SaveLocation(context.Background(), "save_location", tt.input)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to save location with error: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
savedData, err := h.accountFileHandler.ReadAccountData()
|
||||||
|
if err == nil {
|
||||||
|
//Assert that the input provided is what was saved into the file
|
||||||
|
assert.Equal(t, string(tt.input), savedData["Location"])
|
||||||
|
}
|
||||||
|
|
||||||
|
// Assert the results
|
||||||
|
assert.Equal(t, tt.expectedResult, result)
|
||||||
|
assert.Equal(t, tt.expectedError, err)
|
||||||
|
|
||||||
|
// Assert all expectations were met
|
||||||
|
mockFileHandler.AssertExpectations(t)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSaveFirstname(t *testing.T) {
|
||||||
|
// Create a new instance of MockAccountFileHandler
|
||||||
|
mockFileHandler := new(mocks.MockAccountFileHandler)
|
||||||
|
|
||||||
|
// Define test cases
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
input []byte
|
||||||
|
existingData map[string]string
|
||||||
|
writeError error
|
||||||
|
expectedResult resource.Result
|
||||||
|
expectedError error
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "Successful Save",
|
||||||
|
input: []byte("Joe"),
|
||||||
|
existingData: map[string]string{"Name": "Joe"},
|
||||||
|
writeError: nil,
|
||||||
|
expectedResult: resource.Result{},
|
||||||
|
expectedError: nil,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Empty Input",
|
||||||
|
input: []byte{},
|
||||||
|
existingData: map[string]string{"OtherKey": "OtherValue"},
|
||||||
|
writeError: nil,
|
||||||
|
expectedError: nil,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
// Set up the mock expectations
|
||||||
|
mockFileHandler.On("ReadAccountData").Return(tt.existingData, tt.expectedError)
|
||||||
|
if tt.expectedError == nil && len(tt.input) > 0 {
|
||||||
|
mockFileHandler.On("WriteAccountData", mock.MatchedBy(func(data map[string]string) bool {
|
||||||
|
return data["FirstName"] == string(tt.input)
|
||||||
|
})).Return(tt.writeError)
|
||||||
|
} else if len(tt.input) == 0 {
|
||||||
|
// For empty input, no WriteAccountData call should be made
|
||||||
|
mockFileHandler.On("WriteAccountData", mock.Anything).Maybe().Return(tt.writeError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create the Handlers instance with the mock file handler
|
||||||
|
h := &Handlers{
|
||||||
|
accountFileHandler: mockFileHandler,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call save location
|
||||||
|
result, err := h.SaveFirstname(context.Background(), "save_location", tt.input)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to save first name with error: %v", err)
|
||||||
|
}
|
||||||
|
savedData, err := h.accountFileHandler.ReadAccountData()
|
||||||
|
if err == nil {
|
||||||
|
//Assert that the input provided is what was saved into the file
|
||||||
|
assert.Equal(t, string(tt.input), savedData["FirstName"])
|
||||||
|
}
|
||||||
|
|
||||||
|
// Assert the results
|
||||||
|
assert.Equal(t, tt.expectedResult, result)
|
||||||
|
assert.Equal(t, tt.expectedError, err)
|
||||||
|
|
||||||
|
// Assert all expectations were met
|
||||||
|
mockFileHandler.AssertExpectations(t)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSaveFamilyName(t *testing.T) {
|
||||||
|
// Create a new instance of MockAccountFileHandler
|
||||||
|
mockFileHandler := new(mocks.MockAccountFileHandler)
|
||||||
|
|
||||||
|
// Define test cases
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
input []byte
|
||||||
|
existingData map[string]string
|
||||||
|
writeError error
|
||||||
|
expectedResult resource.Result
|
||||||
|
expectedError error
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "Successful Save",
|
||||||
|
input: []byte("Doe"),
|
||||||
|
existingData: map[string]string{"FamilyName": "Doe"},
|
||||||
|
writeError: nil,
|
||||||
|
expectedResult: resource.Result{},
|
||||||
|
expectedError: nil,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Empty Input",
|
||||||
|
input: []byte{},
|
||||||
|
existingData: map[string]string{"FamilyName": "Doe"},
|
||||||
|
writeError: nil,
|
||||||
|
expectedError: nil,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
// Set up the mock expectations
|
||||||
|
mockFileHandler.On("ReadAccountData").Return(tt.existingData, tt.expectedError)
|
||||||
|
if tt.expectedError == nil && len(tt.input) > 0 {
|
||||||
|
mockFileHandler.On("WriteAccountData", mock.MatchedBy(func(data map[string]string) bool {
|
||||||
|
return data["FamilyName"] == string(tt.input)
|
||||||
|
})).Return(tt.writeError)
|
||||||
|
} else if len(tt.input) == 0 {
|
||||||
|
// For empty input, no WriteAccountData call should be made
|
||||||
|
mockFileHandler.On("WriteAccountData", mock.Anything).Maybe().Return(tt.writeError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create the Handlers instance with the mock file handler
|
||||||
|
h := &Handlers{
|
||||||
|
accountFileHandler: mockFileHandler,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call save familyname
|
||||||
|
result, err := h.SaveFamilyname(context.Background(), "save_familyname", tt.input)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to save family name with error: %v", err)
|
||||||
|
}
|
||||||
|
savedData, err := h.accountFileHandler.ReadAccountData()
|
||||||
|
if err == nil {
|
||||||
|
//Assert that the input provided is what was saved into the file
|
||||||
|
assert.Equal(t, string(tt.input), savedData["FamilyName"])
|
||||||
|
}
|
||||||
|
|
||||||
|
// Assert the results
|
||||||
|
assert.Equal(t, tt.expectedResult, result)
|
||||||
|
assert.Equal(t, tt.expectedError, err)
|
||||||
|
|
||||||
|
// Assert all expectations were met
|
||||||
|
mockFileHandler.AssertExpectations(t)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSaveYOB(t *testing.T) {
|
||||||
|
// Create a new instance of MockAccountFileHandler
|
||||||
|
mockFileHandler := new(mocks.MockAccountFileHandler)
|
||||||
|
|
||||||
|
// Define test cases
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
input []byte
|
||||||
|
existingData map[string]string
|
||||||
|
writeError error
|
||||||
|
expectedResult resource.Result
|
||||||
|
expectedError error
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "Successful Save",
|
||||||
|
input: []byte("2006"),
|
||||||
|
existingData: map[string]string{"": ""},
|
||||||
|
writeError: nil,
|
||||||
|
expectedResult: resource.Result{},
|
||||||
|
expectedError: nil,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "YOB less than 4 digits(invalid date entry)",
|
||||||
|
input: []byte{},
|
||||||
|
existingData: map[string]string{"": ""},
|
||||||
|
writeError: nil,
|
||||||
|
expectedError: nil,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
// Set up the mock expectations
|
||||||
|
mockFileHandler.On("ReadAccountData").Return(tt.existingData, tt.expectedError)
|
||||||
|
if tt.expectedError == nil && len(tt.input) > 0 {
|
||||||
|
mockFileHandler.On("WriteAccountData", mock.MatchedBy(func(data map[string]string) bool {
|
||||||
|
return data["YOB"] == string(tt.input)
|
||||||
|
})).Return(tt.writeError)
|
||||||
|
} else if len(tt.input) != 4 {
|
||||||
|
// For input whose input is not a valid yob, no WriteAccountData call should be made
|
||||||
|
mockFileHandler.On("WriteAccountData", mock.Anything).Maybe().Return(tt.writeError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create the Handlers instance with the mock file handler
|
||||||
|
h := &Handlers{
|
||||||
|
accountFileHandler: mockFileHandler,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call save yob
|
||||||
|
result, err := h.SaveYob(context.Background(), "save_yob", tt.input)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to save family name with error: %v", err)
|
||||||
|
}
|
||||||
|
savedData, err := h.accountFileHandler.ReadAccountData()
|
||||||
|
if err == nil {
|
||||||
|
//Assert that the input provided is what was saved into the file
|
||||||
|
assert.Equal(t, string(tt.input), savedData["YOB"])
|
||||||
|
}
|
||||||
|
|
||||||
|
// Assert the results
|
||||||
|
assert.Equal(t, tt.expectedResult, result)
|
||||||
|
assert.Equal(t, tt.expectedError, err)
|
||||||
|
|
||||||
|
// Assert all expectations were met
|
||||||
|
mockFileHandler.AssertExpectations(t)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSaveOfferings(t *testing.T) {
|
||||||
|
// Create a new instance of MockAccountFileHandler
|
||||||
|
mockFileHandler := new(mocks.MockAccountFileHandler)
|
||||||
|
|
||||||
|
// Define test cases
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
input []byte
|
||||||
|
existingData map[string]string
|
||||||
|
writeError error
|
||||||
|
expectedResult resource.Result
|
||||||
|
expectedError error
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "Successful Save",
|
||||||
|
input: []byte("Bananas"),
|
||||||
|
existingData: map[string]string{"": ""},
|
||||||
|
writeError: nil,
|
||||||
|
expectedResult: resource.Result{},
|
||||||
|
expectedError: nil,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Empty input",
|
||||||
|
input: []byte{},
|
||||||
|
existingData: map[string]string{"": ""},
|
||||||
|
writeError: nil,
|
||||||
|
expectedError: nil,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
// Set up the mock expectations
|
||||||
|
mockFileHandler.On("ReadAccountData").Return(tt.existingData, tt.expectedError)
|
||||||
|
if tt.expectedError == nil && len(tt.input) > 0 {
|
||||||
|
mockFileHandler.On("WriteAccountData", mock.MatchedBy(func(data map[string]string) bool {
|
||||||
|
return data["Offerings"] == string(tt.input)
|
||||||
|
})).Return(tt.writeError)
|
||||||
|
} else if len(tt.input) != 4 {
|
||||||
|
// For input whose input is not a valid yob, no WriteAccountData call should be made
|
||||||
|
mockFileHandler.On("WriteAccountData", mock.Anything).Maybe().Return(tt.writeError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create the Handlers instance with the mock file handler
|
||||||
|
h := &Handlers{
|
||||||
|
accountFileHandler: mockFileHandler,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call save yob
|
||||||
|
result, err := h.SaveOfferings(context.Background(), "save_offerings", tt.input)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Failed to save offerings with error: %v", err)
|
||||||
|
}
|
||||||
|
savedData, err := h.accountFileHandler.ReadAccountData()
|
||||||
|
if err == nil {
|
||||||
|
//Assert that the input provided is what was saved into the file
|
||||||
|
assert.Equal(t, string(tt.input), savedData["Offerings"])
|
||||||
|
}
|
||||||
|
|
||||||
|
// Assert the results
|
||||||
|
assert.Equal(t, tt.expectedResult, result)
|
||||||
|
assert.Equal(t, tt.expectedError, err)
|
||||||
|
|
||||||
|
// Assert all expectations were met
|
||||||
|
mockFileHandler.AssertExpectations(t)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSaveGender(t *testing.T) {
|
||||||
|
// Create a new instance of MockAccountFileHandler
|
||||||
|
mockFileHandler := new(mocks.MockAccountFileHandler)
|
||||||
|
|
||||||
|
// Define test cases
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
input []byte
|
||||||
|
existingData map[string]string
|
||||||
|
writeError error
|
||||||
|
expectedResult resource.Result
|
||||||
|
expectedError error
|
||||||
|
expectedGender string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "Successful Save - Male",
|
||||||
|
input: []byte("1"),
|
||||||
|
existingData: map[string]string{"OtherKey": "OtherValue"},
|
||||||
|
writeError: nil,
|
||||||
|
expectedResult: resource.Result{},
|
||||||
|
expectedError: nil,
|
||||||
|
expectedGender: "Male",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Successful Save - Female",
|
||||||
|
input: []byte("2"),
|
||||||
|
existingData: map[string]string{"OtherKey": "OtherValue"},
|
||||||
|
writeError: nil,
|
||||||
|
expectedResult: resource.Result{},
|
||||||
|
expectedError: nil,
|
||||||
|
expectedGender: "Female",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Successful Save - Unspecified",
|
||||||
|
input: []byte("3"),
|
||||||
|
existingData: map[string]string{"OtherKey": "OtherValue"},
|
||||||
|
writeError: nil,
|
||||||
|
expectedResult: resource.Result{},
|
||||||
|
expectedError: nil,
|
||||||
|
expectedGender: "Unspecified",
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
name: "Empty Input",
|
||||||
|
input: []byte{},
|
||||||
|
existingData: map[string]string{"OtherKey": "OtherValue"},
|
||||||
|
writeError: nil,
|
||||||
|
expectedResult: resource.Result{},
|
||||||
|
expectedError: nil,
|
||||||
|
expectedGender: "",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
// Set up the mock expectations
|
||||||
|
mockFileHandler.On("ReadAccountData").Return(tt.existingData, tt.expectedError)
|
||||||
|
if tt.expectedError == nil && len(tt.input) > 0 {
|
||||||
|
mockFileHandler.On("WriteAccountData", mock.MatchedBy(func(data map[string]string) bool {
|
||||||
|
return data["Gender"] == tt.expectedGender
|
||||||
|
})).Return(tt.writeError)
|
||||||
|
} else if len(tt.input) == 0 {
|
||||||
|
// For empty input, no WriteAccountData call should be made
|
||||||
|
mockFileHandler.On("WriteAccountData", mock.Anything).Maybe().Return(tt.writeError)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create the Handlers instance with the mock file handler
|
||||||
|
h := &Handlers{
|
||||||
|
accountFileHandler: mockFileHandler,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call the method
|
||||||
|
result, err := h.SaveGender(context.Background(), "save_gender", tt.input)
|
||||||
|
|
||||||
|
// Assert the results
|
||||||
|
assert.Equal(t, tt.expectedResult, result)
|
||||||
|
assert.Equal(t, tt.expectedError, err)
|
||||||
|
|
||||||
|
// Verify WriteAccountData was called with the expected data
|
||||||
|
if len(tt.input) > 0 && tt.expectedError == nil {
|
||||||
|
mockFileHandler.AssertCalled(t, "WriteAccountData", mock.MatchedBy(func(data map[string]string) bool {
|
||||||
|
return data["Gender"] == tt.expectedGender
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Assert all expectations were met
|
||||||
|
mockFileHandler.AssertExpectations(t)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetSender(t *testing.T) {
|
||||||
|
mockAccountFileHandler := new(mocks.MockAccountFileHandler)
|
||||||
|
h := &Handlers{
|
||||||
|
accountFileHandler: mockAccountFileHandler,
|
||||||
|
}
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
expectedResult resource.Result
|
||||||
|
accountData map[string]string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "Valid public key",
|
||||||
|
expectedResult: resource.Result{
|
||||||
|
Content: "test-public-key",
|
||||||
|
},
|
||||||
|
accountData: map[string]string{
|
||||||
|
"PublicKey": "test-public-key",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Missing public key",
|
||||||
|
expectedResult: resource.Result{
|
||||||
|
Content: "",
|
||||||
|
},
|
||||||
|
accountData: map[string]string{},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
// Reset the mock state
|
||||||
|
mockAccountFileHandler.Mock = mock.Mock{}
|
||||||
|
|
||||||
|
mockAccountFileHandler.On("ReadAccountData").Return(tt.accountData, nil)
|
||||||
|
|
||||||
|
result, err := h.GetSender(context.Background(), "", nil)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Error occurred: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
assert.Equal(t, tt.expectedResult.Content, result.Content)
|
||||||
|
mockAccountFileHandler.AssertCalled(t, "ReadAccountData")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetAmount(t *testing.T) {
|
||||||
|
mockAccountFileHandler := new(mocks.MockAccountFileHandler)
|
||||||
|
h := &Handlers{
|
||||||
|
accountFileHandler: mockAccountFileHandler,
|
||||||
|
}
|
||||||
|
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
expectedResult resource.Result
|
||||||
|
accountData map[string]string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "Valid amount",
|
||||||
|
expectedResult: resource.Result{
|
||||||
|
Content: "0.003",
|
||||||
|
},
|
||||||
|
accountData: map[string]string{
|
||||||
|
"Amount": "0.003",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Missing amount",
|
||||||
|
expectedResult: resource.Result{},
|
||||||
|
accountData: map[string]string{
|
||||||
|
"Amount": "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
// Reset the mock state
|
||||||
|
mockAccountFileHandler.Mock = mock.Mock{}
|
||||||
|
|
||||||
|
mockAccountFileHandler.On("ReadAccountData").Return(tt.accountData, nil)
|
||||||
|
|
||||||
|
result, err := h.GetAmount(context.Background(), "", nil)
|
||||||
|
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, tt.expectedResult.Content, result.Content)
|
||||||
|
|
||||||
|
mockAccountFileHandler.AssertCalled(t, "ReadAccountData")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestGetProfileInfo(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
accountData map[string]string
|
||||||
|
readError error
|
||||||
|
expectedResult resource.Result
|
||||||
|
expectedError error
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "Complete Profile",
|
||||||
|
accountData: map[string]string{
|
||||||
|
"FirstName": "John",
|
||||||
|
"FamilyName": "Doe",
|
||||||
|
"Gender": "Male",
|
||||||
|
"YOB": "1980",
|
||||||
|
"Location": "Mombasa",
|
||||||
|
"Offerings": "Product A",
|
||||||
|
},
|
||||||
|
readError: nil,
|
||||||
|
expectedResult: resource.Result{
|
||||||
|
Content: fmt.Sprintf(
|
||||||
|
"Name: %s %s\nGender: %s\nAge: %d\nLocation: %s\nYou provide: %s\n",
|
||||||
|
"John", "Doe", "Male", 44, "Mombasa", "Product A",
|
||||||
|
),
|
||||||
|
},
|
||||||
|
expectedError: nil,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Profile with Not Provided Fields",
|
||||||
|
accountData: map[string]string{
|
||||||
|
"FirstName": "Not provided",
|
||||||
|
"FamilyName": "Doe",
|
||||||
|
"Gender": "Female",
|
||||||
|
"YOB": "1995",
|
||||||
|
"Location": "Not provided",
|
||||||
|
"Offerings": "Service B",
|
||||||
|
},
|
||||||
|
readError: nil,
|
||||||
|
expectedResult: resource.Result{
|
||||||
|
Content: fmt.Sprintf(
|
||||||
|
"Name: %s\nGender: %s\nAge: %d\nLocation: %s\nYou provide: %s\n",
|
||||||
|
"Not provided", "Female", 29, "Not provided", "Service B",
|
||||||
|
),
|
||||||
|
},
|
||||||
|
expectedError: nil,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Profile with YOB as Not provided",
|
||||||
|
accountData: map[string]string{
|
||||||
|
"FirstName": "Not provided",
|
||||||
|
"FamilyName": "Doe",
|
||||||
|
"Gender": "Female",
|
||||||
|
"YOB": "Not provided",
|
||||||
|
"Location": "Not provided",
|
||||||
|
"Offerings": "Service B",
|
||||||
|
},
|
||||||
|
readError: nil,
|
||||||
|
expectedResult: resource.Result{
|
||||||
|
Content: fmt.Sprintf(
|
||||||
|
"Name: %s\nGender: %s\nAge: %s\nLocation: %s\nYou provide: %s\n",
|
||||||
|
"Not provided", "Female", "Not provided", "Not provided", "Service B",
|
||||||
|
),
|
||||||
|
},
|
||||||
|
expectedError: nil,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
// Create a new instance of MockAccountFileHandler
|
||||||
|
mockFileHandler := new(mocks.MockAccountFileHandler)
|
||||||
|
|
||||||
|
// Set up the mock expectations
|
||||||
|
mockFileHandler.On("ReadAccountData").Return(tt.accountData, tt.readError)
|
||||||
|
|
||||||
|
// Create the Handlers instance with the mock file handler
|
||||||
|
h := &Handlers{
|
||||||
|
accountFileHandler: mockFileHandler,
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call the method
|
||||||
|
result, err := h.GetProfileInfo(context.Background(), "get_profile_info", nil)
|
||||||
|
|
||||||
|
// Assert the results
|
||||||
|
assert.Equal(t, tt.expectedResult, result)
|
||||||
|
assert.Equal(t, tt.expectedError, err)
|
||||||
|
|
||||||
|
// Assert all expectations were met
|
||||||
|
mockFileHandler.AssertExpectations(t)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
44
internal/handlers/ussd/mocks/mocks.go
Normal file
44
internal/handlers/ussd/mocks/mocks.go
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
package mocks
|
||||||
|
|
||||||
|
import (
|
||||||
|
"git.grassecon.net/urdt/ussd/internal/models"
|
||||||
|
"github.com/stretchr/testify/mock"
|
||||||
|
)
|
||||||
|
|
||||||
|
type MockAccountFileHandler struct {
|
||||||
|
mock.Mock
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockAccountFileHandler) EnsureFileExists() error {
|
||||||
|
args := m.Called()
|
||||||
|
return args.Error(0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockAccountFileHandler) ReadAccountData() (map[string]string, error) {
|
||||||
|
args := m.Called()
|
||||||
|
return args.Get(0).(map[string]string), args.Error(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockAccountFileHandler) WriteAccountData(data map[string]string) error {
|
||||||
|
args := m.Called(data)
|
||||||
|
return args.Error(0)
|
||||||
|
}
|
||||||
|
|
||||||
|
type MockAccountService struct {
|
||||||
|
mock.Mock
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockAccountService) CreateAccount() (*models.AccountResponse, error) {
|
||||||
|
args := m.Called()
|
||||||
|
return args.Get(0).(*models.AccountResponse), args.Error(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockAccountService) CheckAccountStatus(TrackingId string) (string, error) {
|
||||||
|
args := m.Called()
|
||||||
|
return args.Get(0).(string), args.Error(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *MockAccountService) CheckBalance(PublicKey string) (string, error) {
|
||||||
|
args := m.Called()
|
||||||
|
return args.Get(0).(string), args.Error(1)
|
||||||
|
}
|
15
internal/models/accountresponse.go
Normal file
15
internal/models/accountresponse.go
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
package models
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
|
||||||
|
)
|
||||||
|
|
||||||
|
type AccountResponse struct {
|
||||||
|
Ok bool `json:"ok"`
|
||||||
|
Result struct {
|
||||||
|
CustodialId json.Number `json:"custodialId"`
|
||||||
|
PublicKey string `json:"publicKey"`
|
||||||
|
TrackingId string `json:"trackingId"`
|
||||||
|
} `json:"result"`
|
||||||
|
}
|
12
internal/models/balanceresponse.go
Normal file
12
internal/models/balanceresponse.go
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
package models
|
||||||
|
|
||||||
|
import "encoding/json"
|
||||||
|
|
||||||
|
|
||||||
|
type BalanceResponse struct {
|
||||||
|
Ok bool `json:"ok"`
|
||||||
|
Result struct {
|
||||||
|
Balance string `json:"balance"`
|
||||||
|
Nonce json.Number `json:"nonce"`
|
||||||
|
} `json:"result"`
|
||||||
|
}
|
20
internal/models/trackstatusresponse.go
Normal file
20
internal/models/trackstatusresponse.go
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
package models
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
type TrackStatusResponse struct {
|
||||||
|
Ok bool `json:"ok"`
|
||||||
|
Result struct {
|
||||||
|
Transaction struct {
|
||||||
|
CreatedAt time.Time `json:"createdAt"`
|
||||||
|
Status string `json:"status"`
|
||||||
|
TransferValue json.Number `json:"transferValue"`
|
||||||
|
TxHash string `json:"txHash"`
|
||||||
|
TxType string `json:"txType"`
|
||||||
|
}
|
||||||
|
} `json:"result"`
|
||||||
|
}
|
46
internal/utils/account_utils.go
Normal file
46
internal/utils/account_utils.go
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
package utils
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"os"
|
||||||
|
)
|
||||||
|
|
||||||
|
type AccountFileHandler struct {
|
||||||
|
FilePath string
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewAccountFileHandler(path string) *AccountFileHandler {
|
||||||
|
return &AccountFileHandler{FilePath: path}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (afh *AccountFileHandler) ReadAccountData() (map[string]string, error) {
|
||||||
|
jsonData, err := os.ReadFile(afh.FilePath)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var accountData map[string]string
|
||||||
|
err = json.Unmarshal(jsonData, &accountData)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return accountData, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (afh *AccountFileHandler) WriteAccountData(accountData map[string]string) error {
|
||||||
|
jsonData, err := json.Marshal(accountData)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return os.WriteFile(afh.FilePath, jsonData, 0644)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (afh *AccountFileHandler) EnsureFileExists() error {
|
||||||
|
f, err := os.OpenFile(afh.FilePath, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return f.Close()
|
||||||
|
}
|
35
internal/utils/age.go
Normal file
35
internal/utils/age.go
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
package utils
|
||||||
|
|
||||||
|
import "time"
|
||||||
|
|
||||||
|
// CalculateAge calculates the age based on a given birthdate and the current date in the format dd/mm/yy
|
||||||
|
// It adjusts for cases where the current date is before the birthday in the current year.
|
||||||
|
func CalculateAge(birthdate, today time.Time) int {
|
||||||
|
today = today.In(birthdate.Location())
|
||||||
|
ty, tm, td := today.Date()
|
||||||
|
today = time.Date(ty, tm, td, 0, 0, 0, 0, time.UTC)
|
||||||
|
by, bm, bd := birthdate.Date()
|
||||||
|
birthdate = time.Date(by, bm, bd, 0, 0, 0, 0, time.UTC)
|
||||||
|
if today.Before(birthdate) {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
age := ty - by
|
||||||
|
anniversary := birthdate.AddDate(age, 0, 0)
|
||||||
|
if anniversary.After(today) {
|
||||||
|
age--
|
||||||
|
}
|
||||||
|
return age
|
||||||
|
}
|
||||||
|
|
||||||
|
// CalculateAgeWithYOB calculates the age based on the given year of birth (YOB).
|
||||||
|
// It subtracts the YOB from the current year to determine the age.
|
||||||
|
//
|
||||||
|
// Parameters:
|
||||||
|
// yob: The year of birth as an integer.
|
||||||
|
//
|
||||||
|
// Returns:
|
||||||
|
// The calculated age as an integer.
|
||||||
|
func CalculateAgeWithYOB(yob int) int {
|
||||||
|
currentYear := time.Now().Year()
|
||||||
|
return currentYear - yob
|
||||||
|
}
|
13
internal/utils/filehandler.go
Normal file
13
internal/utils/filehandler.go
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
package utils
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
type AccountFileHandlerInterface interface {
|
||||||
|
EnsureFileExists() error
|
||||||
|
ReadAccountData() (map[string]string, error)
|
||||||
|
WriteAccountData(data map[string]string) error
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
112
request/base.go
112
request/base.go
@ -1,112 +0,0 @@
|
|||||||
package request
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
|
|
||||||
"git.defalsify.org/vise.git/db"
|
|
||||||
"git.defalsify.org/vise.git/engine"
|
|
||||||
"git.defalsify.org/vise.git/persist"
|
|
||||||
"git.defalsify.org/vise.git/resource"
|
|
||||||
"git.grassecon.net/grassrootseconomics/visedriver/entry"
|
|
||||||
"git.grassecon.net/grassrootseconomics/visedriver/errors"
|
|
||||||
"git.grassecon.net/grassrootseconomics/visedriver/storage"
|
|
||||||
)
|
|
||||||
|
|
||||||
type BaseRequestHandler struct {
|
|
||||||
cfgTemplate engine.Config
|
|
||||||
rp RequestParser
|
|
||||||
rs resource.Resource
|
|
||||||
hn entry.EntryHandler
|
|
||||||
provider storage.StorageProvider
|
|
||||||
}
|
|
||||||
|
|
||||||
// func NewBaseRequestHandler(cfg engine.Config, rs resource.Resource, stateDb db.Db, userdataDb db.Db, rp request.RequestParser, hn *handlers.Handlers) *BaseRequestHandler {
|
|
||||||
func NewBaseRequestHandler(cfg engine.Config, rs resource.Resource, stateDb db.Db, userdataDb db.Db, rp RequestParser, hn entry.EntryHandler) *BaseRequestHandler {
|
|
||||||
return &BaseRequestHandler{
|
|
||||||
cfgTemplate: cfg,
|
|
||||||
rs: rs,
|
|
||||||
hn: hn,
|
|
||||||
rp: rp,
|
|
||||||
provider: storage.NewSimpleStorageProvider(stateDb, userdataDb),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *BaseRequestHandler) Shutdown(ctx context.Context) {
|
|
||||||
err := f.provider.Close(ctx)
|
|
||||||
if err != nil {
|
|
||||||
logg.Errorf("handler shutdown error", "err", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *BaseRequestHandler) GetEngine(cfg engine.Config, rs resource.Resource, pr *persist.Persister) engine.Engine {
|
|
||||||
en := engine.NewEngine(cfg, rs)
|
|
||||||
en = en.WithPersister(pr)
|
|
||||||
return en
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *BaseRequestHandler) Process(rqs RequestSession) (RequestSession, error) {
|
|
||||||
var r bool
|
|
||||||
var err error
|
|
||||||
var ok bool
|
|
||||||
|
|
||||||
logg.InfoCtxf(rqs.Ctx, "new request", "data", rqs)
|
|
||||||
|
|
||||||
rqs.Storage, err = f.provider.Get(rqs.Ctx, rqs.Config.SessionId)
|
|
||||||
if err != nil {
|
|
||||||
logg.ErrorCtxf(rqs.Ctx, "", "storage get error", err)
|
|
||||||
return rqs, errors.ErrStorage
|
|
||||||
}
|
|
||||||
|
|
||||||
//f.hn = f.hn.WithPersister(rqs.Storage.Persister)
|
|
||||||
f.hn.SetPersister(rqs.Storage.Persister)
|
|
||||||
defer func() {
|
|
||||||
f.hn.Exit()
|
|
||||||
}()
|
|
||||||
eni := f.GetEngine(rqs.Config, f.rs, rqs.Storage.Persister)
|
|
||||||
en, ok := eni.(*engine.DefaultEngine)
|
|
||||||
if !ok {
|
|
||||||
perr := f.provider.Put(rqs.Ctx, rqs.Config.SessionId, rqs.Storage)
|
|
||||||
rqs.Storage = nil
|
|
||||||
if perr != nil {
|
|
||||||
logg.ErrorCtxf(rqs.Ctx, "", "storage put error", perr)
|
|
||||||
}
|
|
||||||
return rqs, errors.ErrEngineType
|
|
||||||
}
|
|
||||||
en = en.WithFirst(f.hn.Init)
|
|
||||||
if rqs.Config.EngineDebug {
|
|
||||||
en = en.WithDebug(nil)
|
|
||||||
}
|
|
||||||
rqs.Engine = en
|
|
||||||
|
|
||||||
r, err = rqs.Engine.Exec(rqs.Ctx, rqs.Input)
|
|
||||||
if err != nil {
|
|
||||||
perr := f.provider.Put(rqs.Ctx, rqs.Config.SessionId, rqs.Storage)
|
|
||||||
rqs.Storage = nil
|
|
||||||
if perr != nil {
|
|
||||||
logg.ErrorCtxf(rqs.Ctx, "", "storage put error", perr)
|
|
||||||
}
|
|
||||||
return rqs, err
|
|
||||||
}
|
|
||||||
|
|
||||||
rqs.Continue = r
|
|
||||||
return rqs, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *BaseRequestHandler) Output(rqs RequestSession) (RequestSession, error) {
|
|
||||||
var err error
|
|
||||||
_, err = rqs.Engine.Flush(rqs.Ctx, rqs.Writer)
|
|
||||||
return rqs, err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *BaseRequestHandler) Reset(ctx context.Context, rqs RequestSession) (RequestSession, error) {
|
|
||||||
defer f.provider.Put(ctx, rqs.Config.SessionId, rqs.Storage)
|
|
||||||
return rqs, rqs.Engine.Finish(ctx)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *BaseRequestHandler) GetConfig() engine.Config {
|
|
||||||
return f.cfgTemplate
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *BaseRequestHandler) GetRequestParser() RequestParser {
|
|
||||||
return f.rp
|
|
||||||
}
|
|
@ -1,37 +0,0 @@
|
|||||||
package http
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"io/ioutil"
|
|
||||||
"net/http"
|
|
||||||
|
|
||||||
"git.grassecon.net/grassrootseconomics/visedriver/errors"
|
|
||||||
)
|
|
||||||
|
|
||||||
type DefaultRequestParser struct {
|
|
||||||
}
|
|
||||||
|
|
||||||
func (rp *DefaultRequestParser) GetSessionId(ctx context.Context, rq any) (string, error) {
|
|
||||||
rqv, ok := rq.(*http.Request)
|
|
||||||
if !ok {
|
|
||||||
return "", errors.ErrInvalidRequest
|
|
||||||
}
|
|
||||||
v := rqv.Header.Get("X-Vise-Session")
|
|
||||||
if v == "" {
|
|
||||||
return "", errors.ErrSessionMissing
|
|
||||||
}
|
|
||||||
return v, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (rp *DefaultRequestParser) GetInput(rq any) ([]byte, error) {
|
|
||||||
rqv, ok := rq.(*http.Request)
|
|
||||||
if !ok {
|
|
||||||
return nil, errors.ErrInvalidRequest
|
|
||||||
}
|
|
||||||
defer rqv.Body.Close()
|
|
||||||
v, err := ioutil.ReadAll(rqv.Body)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
return v, nil
|
|
||||||
}
|
|
@ -1,92 +0,0 @@
|
|||||||
package http
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net/http"
|
|
||||||
"strconv"
|
|
||||||
|
|
||||||
"git.defalsify.org/vise.git/logging"
|
|
||||||
"git.grassecon.net/grassrootseconomics/visedriver/errors"
|
|
||||||
"git.grassecon.net/grassrootseconomics/visedriver/request"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
logg = logging.NewVanilla().WithDomain("visedriver.http.session")
|
|
||||||
)
|
|
||||||
|
|
||||||
// HTTPRequestHandler implements the session handler for HTTP
|
|
||||||
type HTTPRequestHandler struct {
|
|
||||||
request.RequestHandler
|
|
||||||
}
|
|
||||||
|
|
||||||
func (f *HTTPRequestHandler) WriteError(w http.ResponseWriter, code int, err error) {
|
|
||||||
s := err.Error()
|
|
||||||
w.Header().Set("Content-Length", strconv.Itoa(len(s)))
|
|
||||||
w.WriteHeader(code)
|
|
||||||
_, err = w.Write([]byte(s))
|
|
||||||
if err != nil {
|
|
||||||
logg.Errorf("error writing error!!", "err", err, "olderr", s)
|
|
||||||
w.WriteHeader(500)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewHTTPRequestHandler(h request.RequestHandler) *HTTPRequestHandler {
|
|
||||||
return &HTTPRequestHandler{
|
|
||||||
RequestHandler: h,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (hh *HTTPRequestHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) {
|
|
||||||
var code int
|
|
||||||
var err error
|
|
||||||
var perr error
|
|
||||||
|
|
||||||
rqs := request.RequestSession{
|
|
||||||
Ctx: req.Context(),
|
|
||||||
Writer: w,
|
|
||||||
}
|
|
||||||
|
|
||||||
rp := hh.GetRequestParser()
|
|
||||||
cfg := hh.GetConfig()
|
|
||||||
cfg.SessionId, err = rp.GetSessionId(req.Context(), req)
|
|
||||||
if err != nil {
|
|
||||||
logg.ErrorCtxf(rqs.Ctx, "", "header processing error", err)
|
|
||||||
hh.WriteError(w, 400, err)
|
|
||||||
}
|
|
||||||
rqs.Config = cfg
|
|
||||||
rqs.Input, err = rp.GetInput(req)
|
|
||||||
if err != nil {
|
|
||||||
logg.ErrorCtxf(rqs.Ctx, "", "header processing error", err)
|
|
||||||
hh.WriteError(w, 400, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
rqs, err = hh.Process(rqs)
|
|
||||||
switch err {
|
|
||||||
case errors.ErrStorage:
|
|
||||||
code = 500
|
|
||||||
case errors.ErrEngineInit:
|
|
||||||
code = 500
|
|
||||||
case errors.ErrEngineExec:
|
|
||||||
code = 500
|
|
||||||
default:
|
|
||||||
code = 200
|
|
||||||
}
|
|
||||||
|
|
||||||
if code != 200 {
|
|
||||||
hh.WriteError(w, 500, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
w.WriteHeader(200)
|
|
||||||
w.Header().Set("Content-Type", "text/plain")
|
|
||||||
rqs, err = hh.Output(rqs)
|
|
||||||
rqs, perr = hh.Reset(rqs.Ctx, rqs)
|
|
||||||
if err != nil {
|
|
||||||
hh.WriteError(w, 500, err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if perr != nil {
|
|
||||||
hh.WriteError(w, 500, perr)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,233 +0,0 @@
|
|||||||
package http
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bytes"
|
|
||||||
"context"
|
|
||||||
"errors"
|
|
||||||
"net/http"
|
|
||||||
"net/http/httptest"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"git.defalsify.org/vise.git/engine"
|
|
||||||
viseerrors "git.grassecon.net/grassrootseconomics/visedriver/errors"
|
|
||||||
"git.grassecon.net/grassrootseconomics/visedriver/request"
|
|
||||||
"git.grassecon.net/grassrootseconomics/visedriver/testutil/mocks/httpmocks"
|
|
||||||
)
|
|
||||||
|
|
||||||
// invalidRequestType is a custom type to test invalid request scenarios
|
|
||||||
type invalidRequestType struct{}
|
|
||||||
|
|
||||||
// errorReader is a helper type that always returns an error when Read is called
|
|
||||||
type errorReader struct{}
|
|
||||||
|
|
||||||
func (e *errorReader) Read(p []byte) (n int, err error) {
|
|
||||||
return 0, errors.New("read error")
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestRequestHandler_ServeHTTP(t *testing.T) {
|
|
||||||
tests := []struct {
|
|
||||||
name string
|
|
||||||
sessionID string
|
|
||||||
input []byte
|
|
||||||
parserErr error
|
|
||||||
processErr error
|
|
||||||
outputErr error
|
|
||||||
resetErr error
|
|
||||||
expectedStatus int
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
name: "Success",
|
|
||||||
sessionID: "123",
|
|
||||||
input: []byte("test input"),
|
|
||||||
expectedStatus: http.StatusOK,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Missing Session ID",
|
|
||||||
sessionID: "",
|
|
||||||
parserErr: viseerrors.ErrSessionMissing,
|
|
||||||
expectedStatus: http.StatusBadRequest,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Process Error",
|
|
||||||
sessionID: "123",
|
|
||||||
input: []byte("test input"),
|
|
||||||
processErr: viseerrors.ErrStorage,
|
|
||||||
expectedStatus: http.StatusInternalServerError,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Output Error",
|
|
||||||
sessionID: "123",
|
|
||||||
input: []byte("test input"),
|
|
||||||
outputErr: errors.New("output error"),
|
|
||||||
expectedStatus: http.StatusOK,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Reset Error",
|
|
||||||
sessionID: "123",
|
|
||||||
input: []byte("test input"),
|
|
||||||
resetErr: errors.New("reset error"),
|
|
||||||
expectedStatus: http.StatusOK,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, tt := range tests {
|
|
||||||
t.Run(tt.name, func(t *testing.T) {
|
|
||||||
mockRequestParser := &httpmocks.MockRequestParser{
|
|
||||||
GetSessionIdFunc: func(any) (string, error) {
|
|
||||||
return tt.sessionID, tt.parserErr
|
|
||||||
},
|
|
||||||
GetInputFunc: func(any) ([]byte, error) {
|
|
||||||
return tt.input, nil
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
mockRequestHandler := &httpmocks.MockRequestHandler{
|
|
||||||
ProcessFunc: func(rs request.RequestSession) (request.RequestSession, error) {
|
|
||||||
return rs, tt.processErr
|
|
||||||
},
|
|
||||||
OutputFunc: func(rs request.RequestSession) (request.RequestSession, error) {
|
|
||||||
return rs, tt.outputErr
|
|
||||||
},
|
|
||||||
ResetFunc: func(ctx context.Context, rs request.RequestSession) (request.RequestSession, error) {
|
|
||||||
return rs, tt.resetErr
|
|
||||||
},
|
|
||||||
GetRequestParserFunc: func() request.RequestParser {
|
|
||||||
return mockRequestParser
|
|
||||||
},
|
|
||||||
GetConfigFunc: func() engine.Config {
|
|
||||||
return engine.Config{}
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
sessionHandler := &HTTPRequestHandler{
|
|
||||||
RequestHandler: mockRequestHandler,
|
|
||||||
}
|
|
||||||
|
|
||||||
req := httptest.NewRequest(http.MethodPost, "/", bytes.NewBuffer(tt.input))
|
|
||||||
req.Header.Set("X-Vise-Session", tt.sessionID)
|
|
||||||
|
|
||||||
rr := httptest.NewRecorder()
|
|
||||||
|
|
||||||
sessionHandler.ServeHTTP(rr, req)
|
|
||||||
|
|
||||||
if status := rr.Code; status != tt.expectedStatus {
|
|
||||||
t.Errorf("handler returned wrong status code: got %v want %v",
|
|
||||||
status, tt.expectedStatus)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestRequestHandler_WriteError(t *testing.T) {
|
|
||||||
handler := &HTTPRequestHandler{}
|
|
||||||
mockWriter := &httpmocks.MockWriter{}
|
|
||||||
err := errors.New("test error")
|
|
||||||
|
|
||||||
handler.WriteError(mockWriter, http.StatusBadRequest, err)
|
|
||||||
|
|
||||||
if mockWriter.WrittenString != "" {
|
|
||||||
t.Errorf("Expected empty body, got %s", mockWriter.WrittenString)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestDefaultRequestParser_GetSessionId(t *testing.T) {
|
|
||||||
tests := []struct {
|
|
||||||
name string
|
|
||||||
request any
|
|
||||||
expectedID string
|
|
||||||
expectedError error
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
name: "Valid Session ID",
|
|
||||||
request: func() *http.Request {
|
|
||||||
req := httptest.NewRequest(http.MethodPost, "/", nil)
|
|
||||||
req.Header.Set("X-Vise-Session", "123456")
|
|
||||||
return req
|
|
||||||
}(),
|
|
||||||
expectedID: "123456",
|
|
||||||
expectedError: nil,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Missing Session ID",
|
|
||||||
request: httptest.NewRequest(http.MethodPost, "/", nil),
|
|
||||||
expectedID: "",
|
|
||||||
expectedError: viseerrors.ErrSessionMissing,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Invalid Request Type",
|
|
||||||
request: invalidRequestType{},
|
|
||||||
expectedID: "",
|
|
||||||
expectedError: viseerrors.ErrInvalidRequest,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
parser := &DefaultRequestParser{}
|
|
||||||
|
|
||||||
for _, tt := range tests {
|
|
||||||
t.Run(tt.name, func(t *testing.T) {
|
|
||||||
id, err := parser.GetSessionId(context.Background(), tt.request)
|
|
||||||
|
|
||||||
if id != tt.expectedID {
|
|
||||||
t.Errorf("Expected session ID %s, got %s", tt.expectedID, id)
|
|
||||||
}
|
|
||||||
|
|
||||||
if err != tt.expectedError {
|
|
||||||
t.Errorf("Expected error %v, got %v", tt.expectedError, err)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestDefaultRequestParser_GetInput(t *testing.T) {
|
|
||||||
tests := []struct {
|
|
||||||
name string
|
|
||||||
request any
|
|
||||||
expectedInput []byte
|
|
||||||
expectedError error
|
|
||||||
}{
|
|
||||||
{
|
|
||||||
name: "Valid Input",
|
|
||||||
request: func() *http.Request {
|
|
||||||
return httptest.NewRequest(http.MethodPost, "/", bytes.NewBufferString("test input"))
|
|
||||||
}(),
|
|
||||||
expectedInput: []byte("test input"),
|
|
||||||
expectedError: nil,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Empty Input",
|
|
||||||
request: httptest.NewRequest(http.MethodPost, "/", nil),
|
|
||||||
expectedInput: []byte{},
|
|
||||||
expectedError: nil,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Invalid Request Type",
|
|
||||||
request: invalidRequestType{},
|
|
||||||
expectedInput: nil,
|
|
||||||
expectedError: viseerrors.ErrInvalidRequest,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: "Read Error",
|
|
||||||
request: func() *http.Request {
|
|
||||||
return httptest.NewRequest(http.MethodPost, "/", &errorReader{})
|
|
||||||
}(),
|
|
||||||
expectedInput: nil,
|
|
||||||
expectedError: errors.New("read error"),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
parser := &DefaultRequestParser{}
|
|
||||||
|
|
||||||
for _, tt := range tests {
|
|
||||||
t.Run(tt.name, func(t *testing.T) {
|
|
||||||
input, err := parser.GetInput(tt.request)
|
|
||||||
|
|
||||||
if !bytes.Equal(input, tt.expectedInput) {
|
|
||||||
t.Errorf("Expected input %s, got %s", tt.expectedInput, input)
|
|
||||||
}
|
|
||||||
|
|
||||||
if err != tt.expectedError && (err == nil || err.Error() != tt.expectedError.Error()) {
|
|
||||||
t.Errorf("Expected error %v, got %v", tt.expectedError, err)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,42 +0,0 @@
|
|||||||
package request
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"io"
|
|
||||||
|
|
||||||
"git.defalsify.org/vise.git/engine"
|
|
||||||
"git.defalsify.org/vise.git/logging"
|
|
||||||
"git.defalsify.org/vise.git/persist"
|
|
||||||
"git.defalsify.org/vise.git/resource"
|
|
||||||
"git.grassecon.net/grassrootseconomics/visedriver/storage"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
logg = logging.NewVanilla().WithDomain("visedriver.request")
|
|
||||||
)
|
|
||||||
|
|
||||||
type RequestSession struct {
|
|
||||||
Ctx context.Context
|
|
||||||
Config engine.Config
|
|
||||||
Engine engine.Engine
|
|
||||||
Input []byte
|
|
||||||
Storage *storage.Storage
|
|
||||||
Writer io.Writer
|
|
||||||
Continue bool
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: seems like can remove this.
|
|
||||||
type RequestParser interface {
|
|
||||||
GetSessionId(context.Context, any) (string, error)
|
|
||||||
GetInput(any) ([]byte, error)
|
|
||||||
}
|
|
||||||
|
|
||||||
type RequestHandler interface {
|
|
||||||
GetConfig() engine.Config
|
|
||||||
GetRequestParser() RequestParser
|
|
||||||
GetEngine(engine.Config, resource.Resource, *persist.Persister) engine.Engine
|
|
||||||
Process(RequestSession) (RequestSession, error)
|
|
||||||
Output(RequestSession) (RequestSession, error)
|
|
||||||
Reset(context.Context, RequestSession) (RequestSession, error)
|
|
||||||
Shutdown(ctx context.Context)
|
|
||||||
}
|
|
@ -1,44 +0,0 @@
|
|||||||
{
|
|
||||||
"ok": true,
|
|
||||||
"description": "Token holdings with current balances",
|
|
||||||
"result": {
|
|
||||||
"holdings": [
|
|
||||||
{
|
|
||||||
"contractAddress": "0x6CC75A06ac72eB4Db2eE22F781F5D100d8ec03ee",
|
|
||||||
"tokenSymbol": "FSPTST",
|
|
||||||
"tokenDecimals": "6",
|
|
||||||
"balance": "8869964242"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"contractAddress": "0x724F2910D790B54A39a7638282a45B1D83564fFA",
|
|
||||||
"tokenSymbol": "GEO",
|
|
||||||
"tokenDecimals": "6",
|
|
||||||
"balance": "9884"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"contractAddress": "0x2105a206B7bec31E2F90acF7385cc8F7F5f9D273",
|
|
||||||
"tokenSymbol": "MFNK",
|
|
||||||
"tokenDecimals": "6",
|
|
||||||
"balance": "19788697"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"contractAddress": "0x63DE2Ac8D1008351Cc69Fb8aCb94Ba47728a7E83",
|
|
||||||
"tokenSymbol": "MILO",
|
|
||||||
"tokenDecimals": "6",
|
|
||||||
"balance": "75"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"contractAddress": "0xd4c288865Ce0985a481Eef3be02443dF5E2e4Ea9",
|
|
||||||
"tokenSymbol": "SOHAIL",
|
|
||||||
"tokenDecimals": "6",
|
|
||||||
"balance": "27874115"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"contractAddress": "0x45d747172e77d55575c197CbA9451bC2CD8F4958",
|
|
||||||
"tokenSymbol": "SRF",
|
|
||||||
"tokenDecimals": "6",
|
|
||||||
"balance": "2745987"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
17
services/registration/Makefile
Normal file
17
services/registration/Makefile
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
# Variables to match files in the current directory
|
||||||
|
INPUTS = $(wildcard ./*.vis)
|
||||||
|
TXTS = $(wildcard ./*.txt.orig)
|
||||||
|
|
||||||
|
# Rule to build .bin files from .vis files
|
||||||
|
%.vis:
|
||||||
|
go run ../../go-vise/dev/asm -f pp.csv $(basename $@).vis > $(basename $@).bin
|
||||||
|
@echo "Built $(basename $@).bin from $(basename $@).vis"
|
||||||
|
|
||||||
|
# Rule to copy .orig files to .txt
|
||||||
|
%.txt.orig:
|
||||||
|
cp -v $(basename $@).orig $(basename $@)
|
||||||
|
@echo "Copied $(basename $@).orig to $(basename $@)"
|
||||||
|
|
||||||
|
# 'all' target depends on all .vis and .txt.orig files
|
||||||
|
all: $(INPUTS) $(TXTS)
|
||||||
|
@echo "Running all: $(INPUTS) $(TXTS)"
|
1
services/registration/account_creation
Normal file
1
services/registration/account_creation
Normal file
@ -0,0 +1 @@
|
|||||||
|
Your account is being created...
|
4
services/registration/account_creation.vis
Normal file
4
services/registration/account_creation.vis
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
RELOAD verify_pin
|
||||||
|
CATCH create_pin_mismatch flag_pin_mismatch 1
|
||||||
|
LOAD quit 0
|
||||||
|
HALT
|
1
services/registration/account_creation_failed
Normal file
1
services/registration/account_creation_failed
Normal file
@ -0,0 +1 @@
|
|||||||
|
Your account creation request failed. Please try again later.
|
3
services/registration/account_creation_failed.vis
Normal file
3
services/registration/account_creation_failed.vis
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
MOUT quit 9
|
||||||
|
HALT
|
||||||
|
INCMP quit 9
|
1
services/registration/account_creation_failed_swa
Normal file
1
services/registration/account_creation_failed_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Ombi lako la kusajiliwa haliwezi kukamilishwa. Tafadhali jaribu tena baadaye.
|
1
services/registration/account_creation_swa
Normal file
1
services/registration/account_creation_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Akaunti yako inatengenezwa...
|
1
services/registration/account_menu
Normal file
1
services/registration/account_menu
Normal file
@ -0,0 +1 @@
|
|||||||
|
My Account
|
1
services/registration/account_menu_swa
Normal file
1
services/registration/account_menu_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Akaunti yangu
|
1
services/registration/account_pending
Normal file
1
services/registration/account_pending
Normal file
@ -0,0 +1 @@
|
|||||||
|
Your account is still being created.
|
3
services/registration/account_pending.vis
Normal file
3
services/registration/account_pending.vis
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
RELOAD check_account_status
|
||||||
|
CATCH main flag_account_success 1
|
||||||
|
HALT
|
1
services/registration/account_pending_swa
Normal file
1
services/registration/account_pending_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Akaunti yako bado inatengenezwa
|
1
services/registration/address
Normal file
1
services/registration/address
Normal file
@ -0,0 +1 @@
|
|||||||
|
Address: {{.check_identifier}}
|
6
services/registration/address.vis
Normal file
6
services/registration/address.vis
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
LOAD check_identifier 0
|
||||||
|
RELOAD check_identifier
|
||||||
|
MAP check_identifier
|
||||||
|
MOUT quit 9
|
||||||
|
HALT
|
||||||
|
INCMP quit 9
|
2
services/registration/amount
Normal file
2
services/registration/amount
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
Maximum amount: {{.max_amount}}
|
||||||
|
Enter amount:
|
12
services/registration/amount.vis
Normal file
12
services/registration/amount.vis
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
LOAD reset_transaction_amount 0
|
||||||
|
LOAD max_amount 10
|
||||||
|
MAP max_amount
|
||||||
|
MOUT back 0
|
||||||
|
HALT
|
||||||
|
LOAD validate_amount 64
|
||||||
|
RELOAD validate_amount
|
||||||
|
CATCH invalid_amount flag_invalid_amount 1
|
||||||
|
INCMP _ 0
|
||||||
|
LOAD get_recipient 12
|
||||||
|
LOAD get_sender 64
|
||||||
|
INCMP transaction_pin *
|
2
services/registration/amount_swa
Normal file
2
services/registration/amount_swa
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
Kiwango cha juu: {{.max_amount}}
|
||||||
|
Weka kiwango:
|
1
services/registration/back_menu
Normal file
1
services/registration/back_menu
Normal file
@ -0,0 +1 @@
|
|||||||
|
Back
|
1
services/registration/back_menu_swa
Normal file
1
services/registration/back_menu_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Rudi
|
1
services/registration/balances
Normal file
1
services/registration/balances
Normal file
@ -0,0 +1 @@
|
|||||||
|
Balances:
|
8
services/registration/balances.vis
Normal file
8
services/registration/balances.vis
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
LOAD reset_account_authorized 0
|
||||||
|
MOUT my_balance 1
|
||||||
|
MOUT community_balance 2
|
||||||
|
MOUT back 0
|
||||||
|
HALT
|
||||||
|
INCMP _ 0
|
||||||
|
INCMP my_balance 1
|
||||||
|
INCMP community_balance 2
|
1
services/registration/balances_swa
Normal file
1
services/registration/balances_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Salio
|
1
services/registration/change_language_menu
Normal file
1
services/registration/change_language_menu
Normal file
@ -0,0 +1 @@
|
|||||||
|
Change language
|
1
services/registration/change_language_menu_swa
Normal file
1
services/registration/change_language_menu_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Badili lugha
|
1
services/registration/change_pin_menu
Normal file
1
services/registration/change_pin_menu
Normal file
@ -0,0 +1 @@
|
|||||||
|
Change PIN
|
1
services/registration/change_pin_menu_swa
Normal file
1
services/registration/change_pin_menu_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Badili PIN
|
1
services/registration/check_balance_menu
Normal file
1
services/registration/check_balance_menu
Normal file
@ -0,0 +1 @@
|
|||||||
|
Check balances
|
1
services/registration/check_balance_menu_swa
Normal file
1
services/registration/check_balance_menu_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Angalia salio
|
1
services/registration/check_statement_menu
Normal file
1
services/registration/check_statement_menu
Normal file
@ -0,0 +1 @@
|
|||||||
|
Check statement
|
1
services/registration/check_statement_menu_swa
Normal file
1
services/registration/check_statement_menu_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Taarifa ya matumizi
|
1
services/registration/comminity_balance_swa
Normal file
1
services/registration/comminity_balance_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Salio la kikundi
|
2
services/registration/community_balance
Normal file
2
services/registration/community_balance
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
Your community balance is: 0.00SRF
|
||||||
|
|
5
services/registration/community_balance.vis
Normal file
5
services/registration/community_balance.vis
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
LOAD reset_incorrect 0
|
||||||
|
CATCH incorrect_pin flag_incorrect_pin 1
|
||||||
|
CATCH pin_entry flag_account_authorized 0
|
||||||
|
LOAD quit_with_balance 0
|
||||||
|
HALT
|
1
services/registration/community_balance_menu
Normal file
1
services/registration/community_balance_menu
Normal file
@ -0,0 +1 @@
|
|||||||
|
Community balance
|
1
services/registration/community_balance_menu_swa
Normal file
1
services/registration/community_balance_menu_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Salio la kikundi
|
1
services/registration/confirm_create_pin
Normal file
1
services/registration/confirm_create_pin
Normal file
@ -0,0 +1 @@
|
|||||||
|
Enter your four number PIN again:
|
4
services/registration/confirm_create_pin.vis
Normal file
4
services/registration/confirm_create_pin.vis
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
LOAD save_pin 0
|
||||||
|
HALT
|
||||||
|
LOAD verify_pin 8
|
||||||
|
INCMP account_creation *
|
1
services/registration/confirm_create_pin_swa
Normal file
1
services/registration/confirm_create_pin_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Weka PIN yako tena:
|
1
services/registration/create_pin
Normal file
1
services/registration/create_pin
Normal file
@ -0,0 +1 @@
|
|||||||
|
Please enter a new four number PIN for your account:
|
9
services/registration/create_pin.vis
Normal file
9
services/registration/create_pin.vis
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
LOAD create_account 0
|
||||||
|
CATCH account_creation_failed flag_account_creation_failed 1
|
||||||
|
MOUT exit 0
|
||||||
|
HALT
|
||||||
|
LOAD save_pin 0
|
||||||
|
RELOAD save_pin
|
||||||
|
CATCH . flag_incorrect_pin 1
|
||||||
|
INCMP quit 0
|
||||||
|
INCMP confirm_create_pin *
|
1
services/registration/create_pin_mismatch
Normal file
1
services/registration/create_pin_mismatch
Normal file
@ -0,0 +1 @@
|
|||||||
|
The PIN is not a match. Try again
|
5
services/registration/create_pin_mismatch.vis
Normal file
5
services/registration/create_pin_mismatch.vis
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
MOUT retry 1
|
||||||
|
MOUT quit 9
|
||||||
|
HALT
|
||||||
|
INCMP confirm_create_pin 1
|
||||||
|
INCMP quit 9
|
1
services/registration/create_pin_mismatch_swa
Normal file
1
services/registration/create_pin_mismatch_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
PIN uliyoweka haifanani. Jaribu tena
|
1
services/registration/create_pin_swa
Normal file
1
services/registration/create_pin_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Tafadhali weka PIN mpya yenye nambari nne kwa akaunti yako:
|
5
services/registration/display_profile_info
Normal file
5
services/registration/display_profile_info
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
Wasifu wangu
|
||||||
|
Name: Not provided
|
||||||
|
Gender: Not provided
|
||||||
|
Age: Not provided
|
||||||
|
Location: Not provided
|
3
services/registration/display_profile_info.vis
Normal file
3
services/registration/display_profile_info.vis
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
MOUT back 0
|
||||||
|
HALT
|
||||||
|
INCMP _ 0
|
0
services/registration/display_profile_info_swa
Normal file
0
services/registration/display_profile_info_swa
Normal file
1
services/registration/edit_gender_menu
Normal file
1
services/registration/edit_gender_menu
Normal file
@ -0,0 +1 @@
|
|||||||
|
Edit gender
|
1
services/registration/edit_gender_menu_swa
Normal file
1
services/registration/edit_gender_menu_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Weka jinsia
|
1
services/registration/edit_location_menu
Normal file
1
services/registration/edit_location_menu
Normal file
@ -0,0 +1 @@
|
|||||||
|
Edit location
|
1
services/registration/edit_location_menu_swa
Normal file
1
services/registration/edit_location_menu_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Weka eneo
|
1
services/registration/edit_name_menu
Normal file
1
services/registration/edit_name_menu
Normal file
@ -0,0 +1 @@
|
|||||||
|
Edit name
|
1
services/registration/edit_name_menu_swa
Normal file
1
services/registration/edit_name_menu_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Weka jina
|
1
services/registration/edit_offerings_menu
Normal file
1
services/registration/edit_offerings_menu
Normal file
@ -0,0 +1 @@
|
|||||||
|
Edit offerings
|
1
services/registration/edit_offerings_menu_swa
Normal file
1
services/registration/edit_offerings_menu_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Weka unachouza
|
1
services/registration/edit_profile
Normal file
1
services/registration/edit_profile
Normal file
@ -0,0 +1 @@
|
|||||||
|
My profile
|
20
services/registration/edit_profile.vis
Normal file
20
services/registration/edit_profile.vis
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
LOAD reset_account_authorized 16
|
||||||
|
LOAD reset_allow_update 0
|
||||||
|
RELOAD reset_allow_update
|
||||||
|
MOUT edit_name 1
|
||||||
|
MOUT edit_gender 2
|
||||||
|
MOUT edit_yob 3
|
||||||
|
MOUT edit_location 4
|
||||||
|
MOUT edit_offerings 5
|
||||||
|
MOUT view 6
|
||||||
|
MOUT back 0
|
||||||
|
HALT
|
||||||
|
INCMP _ 0
|
||||||
|
LOAD set_reset_single_edit 0
|
||||||
|
RELOAD set_reset_single_edit
|
||||||
|
INCMP enter_name 1
|
||||||
|
INCMP select_gender 2
|
||||||
|
INCMP enter_yob 3
|
||||||
|
INCMP enter_location 4
|
||||||
|
INCMP enter_offerings 5
|
||||||
|
INCMP view_profile 6
|
1
services/registration/edit_profile_swa
Normal file
1
services/registration/edit_profile_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Wasifu wangu
|
1
services/registration/edit_yob_menu
Normal file
1
services/registration/edit_yob_menu
Normal file
@ -0,0 +1 @@
|
|||||||
|
Edit year of birth
|
1
services/registration/edit_yob_menu_swa
Normal file
1
services/registration/edit_yob_menu_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Weka mwaka wa kuzaliwa
|
1
services/registration/enter_familyname
Normal file
1
services/registration/enter_familyname
Normal file
@ -0,0 +1 @@
|
|||||||
|
Enter family name:
|
5
services/registration/enter_familyname.vis
Normal file
5
services/registration/enter_familyname.vis
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
LOAD save_firstname 0
|
||||||
|
MOUT back 0
|
||||||
|
HALT
|
||||||
|
INCMP _ 0
|
||||||
|
INCMP select_gender *
|
0
services/registration/enter_familyname_swa
Normal file
0
services/registration/enter_familyname_swa
Normal file
1
services/registration/enter_location
Normal file
1
services/registration/enter_location
Normal file
@ -0,0 +1 @@
|
|||||||
|
Enter your location:
|
9
services/registration/enter_location.vis
Normal file
9
services/registration/enter_location.vis
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
CATCH incorrect_date_format flag_incorrect_date_format 1
|
||||||
|
LOAD save_yob 0
|
||||||
|
CATCH update_success flag_allow_update 1
|
||||||
|
MOUT back 0
|
||||||
|
HALT
|
||||||
|
INCMP _ 0
|
||||||
|
LOAD save_location 0
|
||||||
|
CATCH pin_entry flag_single_edit 1
|
||||||
|
INCMP enter_offerings *
|
1
services/registration/enter_location_swa
Normal file
1
services/registration/enter_location_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Weka eneo:
|
1
services/registration/enter_name
Normal file
1
services/registration/enter_name
Normal file
@ -0,0 +1 @@
|
|||||||
|
Enter your first names:
|
4
services/registration/enter_name.vis
Normal file
4
services/registration/enter_name.vis
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
MOUT back 0
|
||||||
|
HALT
|
||||||
|
INCMP _ 0
|
||||||
|
INCMP enter_familyname *
|
1
services/registration/enter_name_swa
Normal file
1
services/registration/enter_name_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Weka majina yako ya kwanza:
|
1
services/registration/enter_offerings
Normal file
1
services/registration/enter_offerings
Normal file
@ -0,0 +1 @@
|
|||||||
|
Enter the services or goods you offer:
|
8
services/registration/enter_offerings.vis
Normal file
8
services/registration/enter_offerings.vis
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
LOAD save_location 0
|
||||||
|
CATCH incorrect_pin flag_incorrect_pin 1
|
||||||
|
CATCH update_success flag_allow_update 1
|
||||||
|
MOUT back 0
|
||||||
|
HALT
|
||||||
|
LOAD save_offerings 0
|
||||||
|
INCMP _ 0
|
||||||
|
INCMP pin_entry *
|
1
services/registration/enter_offerings_swa
Normal file
1
services/registration/enter_offerings_swa
Normal file
@ -0,0 +1 @@
|
|||||||
|
Weka unachouza
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user