diff --git a/coverage.html b/coverage.html
deleted file mode 100644
index a448bc0..0000000
--- a/coverage.html
+++ /dev/null
@@ -1,2961 +0,0 @@
-
-
-
-
-
-
-
package main
-
-import (
- "context"
- "flag"
- "fmt"
- "net/http"
- "os"
- "os/signal"
- "path"
- "strconv"
- "strings"
- "syscall"
-
- "git.defalsify.org/vise.git/engine"
- "git.defalsify.org/vise.git/logging"
- "git.defalsify.org/vise.git/resource"
-
- "git.grassecon.net/urdt/ussd/internal/handlers"
- httpserver "git.grassecon.net/urdt/ussd/internal/http"
- "git.grassecon.net/urdt/ussd/internal/storage"
-)
-
-var (
- logg = logging.NewVanilla()
- scriptDir = path.Join("services", "registration")
-)
-
-type atRequestParser struct{}
-
-func (arp *atRequestParser) GetSessionId(rq any) (string, error) {
- rqv, ok := rq.(*http.Request)
- if !ok {
- return "", handlers.ErrInvalidRequest
- }
- if err := rqv.ParseForm(); err != nil {
- return "", fmt.Errorf("failed to parse form data: %v", err)
- }
-
- phoneNumber := rqv.FormValue("phoneNumber")
- if phoneNumber == "" {
- return "", fmt.Errorf("no phone number found")
- }
-
- return phoneNumber, nil
-}
-
-func (arp *atRequestParser) GetInput(rq any) ([]byte, error) {
- rqv, ok := rq.(*http.Request)
- if !ok {
- return nil, handlers.ErrInvalidRequest
- }
- if err := rqv.ParseForm(); err != nil {
- return nil, fmt.Errorf("failed to parse form data: %v", err)
- }
-
- text := rqv.FormValue("text")
-
- parts := strings.Split(text, "*")
- if len(parts) == 0 {
- return nil, fmt.Errorf("no input found")
- }
-
- return []byte(parts[len(parts)-1]), nil
-}
-
-func main() {
- var dbDir string
- var resourceDir string
- var size uint
- var engineDebug bool
- var host string
- var port uint
- flag.StringVar(&dbDir, "dbdir", ".state", "database dir to read from")
- flag.StringVar(&resourceDir, "resourcedir", path.Join("services", "registration"), "resource dir")
- flag.BoolVar(&engineDebug, "d", false, "use engine debug output")
- flag.UintVar(&size, "s", 160, "max size of output")
- flag.StringVar(&host, "h", "127.0.0.1", "http host")
- flag.UintVar(&port, "p", 7123, "http port")
- flag.Parse()
-
- logg.Infof("start command", "dbdir", dbDir, "resourcedir", resourceDir, "outputsize", size)
-
- ctx := context.Background()
- pfp := path.Join(scriptDir, "pp.csv")
-
- cfg := engine.Config{
- Root: "root",
- OutputSize: uint32(size),
- FlagCount: uint32(20),
- }
-
- if engineDebug {
- cfg.EngineDebug = true
- }
-
- menuStorageService := storage.NewMenuStorageService(dbDir, resourceDir)
- rs, err := menuStorageService.GetResource(ctx)
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- err = menuStorageService.EnsureDbDir()
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- userdataStore, err := menuStorageService.GetUserdataDb(ctx)
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
- defer userdataStore.Close()
-
- dbResource, ok := rs.(*resource.DbResource)
- if !ok {
- os.Exit(1)
- }
-
- lhs, err := handlers.NewLocalHandlerService(pfp, true, dbResource, cfg, rs)
- lhs.SetDataStore(&userdataStore)
-
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- hl, err := lhs.GetHandler()
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- stateStore, err := menuStorageService.GetStateStore(ctx)
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
- defer stateStore.Close()
-
- rp := &atRequestParser{}
- bsh := handlers.NewBaseSessionHandler(cfg, rs, stateStore, userdataStore, rp, hl)
- sh := httpserver.NewATSessionHandler(bsh)
- s := &http.Server{
- Addr: fmt.Sprintf("%s:%s", host, strconv.Itoa(int(port))),
- Handler: sh,
- }
- s.RegisterOnShutdown(sh.Shutdown)
-
- cint := make(chan os.Signal)
- cterm := make(chan os.Signal)
- signal.Notify(cint, os.Interrupt, syscall.SIGINT)
- signal.Notify(cterm, os.Interrupt, syscall.SIGTERM)
- go func() {
- select {
- case _ = <-cint:
- case _ = <-cterm:
- }
- s.Shutdown(ctx)
- }()
- err = s.ListenAndServe()
- if err != nil {
- logg.Infof("Server closed with error", "err", err)
- }
-}
-
-
-
package main
-
-import (
- "context"
- "flag"
- "fmt"
- "os"
- "os/signal"
- "path"
- "syscall"
-
- "git.defalsify.org/vise.git/engine"
- "git.defalsify.org/vise.git/logging"
- "git.defalsify.org/vise.git/resource"
-
- "git.grassecon.net/urdt/ussd/internal/handlers"
- "git.grassecon.net/urdt/ussd/internal/storage"
-)
-
-var (
- logg = logging.NewVanilla()
- scriptDir = path.Join("services", "registration")
-)
-
-type asyncRequestParser struct {
- sessionId string
- input []byte
-}
-
-func (p *asyncRequestParser) GetSessionId(r any) (string, error) {
- return p.sessionId, nil
-}
-
-func (p *asyncRequestParser) GetInput(r any) ([]byte, error) {
- return p.input, nil
-}
-
-func main() {
- var sessionId string
- var dbDir string
- var resourceDir string
- var size uint
- var engineDebug bool
- var host string
- var port uint
- flag.StringVar(&sessionId, "session-id", "075xx2123", "session id")
- flag.StringVar(&dbDir, "dbdir", ".state", "database dir to read from")
- flag.StringVar(&resourceDir, "resourcedir", path.Join("services", "registration"), "resource dir")
- flag.BoolVar(&engineDebug, "d", false, "use engine debug output")
- flag.UintVar(&size, "s", 160, "max size of output")
- flag.StringVar(&host, "h", "127.0.0.1", "http host")
- flag.UintVar(&port, "p", 7123, "http port")
- flag.Parse()
-
- logg.Infof("start command", "dbdir", dbDir, "resourcedir", resourceDir, "outputsize", size, "sessionId", sessionId)
-
- ctx := context.Background()
- pfp := path.Join(scriptDir, "pp.csv")
-
- cfg := engine.Config{
- Root: "root",
- OutputSize: uint32(size),
- FlagCount: uint32(16),
- }
-
- if engineDebug {
- cfg.EngineDebug = true
- }
-
- menuStorageService := storage.NewMenuStorageService(dbDir, resourceDir)
- rs, err := menuStorageService.GetResource(ctx)
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- err = menuStorageService.EnsureDbDir()
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- userdataStore, err := menuStorageService.GetUserdataDb(ctx)
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
- defer userdataStore.Close()
-
- dbResource, ok := rs.(*resource.DbResource)
- if !ok {
- os.Exit(1)
- }
-
- lhs, err := handlers.NewLocalHandlerService(pfp, true, dbResource, cfg, rs)
- lhs.SetDataStore(&userdataStore)
-
- hl, err := lhs.GetHandler()
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- stateStore, err := menuStorageService.GetStateStore(ctx)
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
- defer stateStore.Close()
-
- rp := &asyncRequestParser{
- sessionId: sessionId,
- }
- sh := handlers.NewBaseSessionHandler(cfg, rs, stateStore, userdataStore, rp, hl)
- cfg.SessionId = sessionId
- rqs := handlers.RequestSession{
- Ctx: ctx,
- Writer: os.Stdout,
- Config: cfg,
- }
-
- cint := make(chan os.Signal)
- cterm := make(chan os.Signal)
- signal.Notify(cint, os.Interrupt, syscall.SIGINT)
- signal.Notify(cterm, os.Interrupt, syscall.SIGTERM)
- go func() {
- select {
- case _ = <-cint:
- case _ = <-cterm:
- }
- sh.Shutdown()
- }()
-
- for true {
- rqs, err = sh.Process(rqs)
- if err != nil {
- logg.ErrorCtxf(ctx, "error in process: %v", "err", err)
- fmt.Errorf("error in process: %v", err)
- os.Exit(1)
- }
- rqs, err = sh.Output(rqs)
- if err != nil {
- logg.ErrorCtxf(ctx, "error in output: %v", "err", err)
- fmt.Errorf("error in output: %v", err)
- os.Exit(1)
- }
- rqs, err = sh.Reset(rqs)
- if err != nil {
- logg.ErrorCtxf(ctx, "error in reset: %v", "err", err)
- fmt.Errorf("error in reset: %v", err)
- os.Exit(1)
- }
- fmt.Println("")
- _, err = fmt.Scanln(&rqs.Input)
- if err != nil {
- logg.ErrorCtxf(ctx, "error in input", "err", err)
- fmt.Errorf("error in input: %v", err)
- os.Exit(1)
- }
- }
-}
-
-
-
package main
-
-import (
- "context"
- "flag"
- "fmt"
- "net/http"
- "os"
- "os/signal"
- "path"
- "strconv"
- "syscall"
-
- "git.defalsify.org/vise.git/engine"
- "git.defalsify.org/vise.git/logging"
- "git.defalsify.org/vise.git/resource"
-
- "git.grassecon.net/urdt/ussd/internal/handlers"
- httpserver "git.grassecon.net/urdt/ussd/internal/http"
- "git.grassecon.net/urdt/ussd/internal/storage"
-)
-
-var (
- logg = logging.NewVanilla()
- scriptDir = path.Join("services", "registration")
-)
-
-func main() {
- var dbDir string
- var resourceDir string
- var size uint
- var engineDebug bool
- var host string
- var port uint
- flag.StringVar(&dbDir, "dbdir", ".state", "database dir to read from")
- flag.StringVar(&resourceDir, "resourcedir", path.Join("services", "registration"), "resource dir")
- flag.BoolVar(&engineDebug, "d", false, "use engine debug output")
- flag.UintVar(&size, "s", 160, "max size of output")
- flag.StringVar(&host, "h", "127.0.0.1", "http host")
- flag.UintVar(&port, "p", 7123, "http port")
- flag.Parse()
-
- logg.Infof("start command", "dbdir", dbDir, "resourcedir", resourceDir, "outputsize", size)
-
- ctx := context.Background()
- pfp := path.Join(scriptDir, "pp.csv")
-
- cfg := engine.Config{
- Root: "root",
- OutputSize: uint32(size),
- FlagCount: uint32(16),
- }
-
- if engineDebug {
- cfg.EngineDebug = true
- }
-
- menuStorageService := storage.NewMenuStorageService(dbDir, resourceDir)
- rs, err := menuStorageService.GetResource(ctx)
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- err = menuStorageService.EnsureDbDir()
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- userdataStore, err := menuStorageService.GetUserdataDb(ctx)
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
- defer userdataStore.Close()
-
- dbResource, ok := rs.(*resource.DbResource)
- if !ok {
- os.Exit(1)
- }
-
- lhs, err := handlers.NewLocalHandlerService(pfp, true, dbResource, cfg, rs)
- lhs.SetDataStore(&userdataStore)
-
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- hl, err := lhs.GetHandler()
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- stateStore, err := menuStorageService.GetStateStore(ctx)
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
- defer stateStore.Close()
-
- rp := &httpserver.DefaultRequestParser{}
- bsh := handlers.NewBaseSessionHandler(cfg, rs, stateStore, userdataStore, rp, hl)
- sh := httpserver.ToSessionHandler(bsh)
- s := &http.Server{
- Addr: fmt.Sprintf("%s:%s", host, strconv.Itoa(int(port))),
- Handler: sh,
- }
- s.RegisterOnShutdown(sh.Shutdown)
-
- cint := make(chan os.Signal)
- cterm := make(chan os.Signal)
- signal.Notify(cint, os.Interrupt, syscall.SIGINT)
- signal.Notify(cterm, os.Interrupt, syscall.SIGTERM)
- go func() {
- select {
- case _ = <-cint:
- case _ = <-cterm:
- }
- s.Shutdown(ctx)
- }()
- err = s.ListenAndServe()
- if err != nil {
- logg.Infof("Server closed with error", "err", err)
- }
-}
-
-
-
package main
-
-import (
- "context"
- "flag"
- "fmt"
- "os"
- "path"
-
- "git.defalsify.org/vise.git/engine"
- "git.defalsify.org/vise.git/logging"
- "git.defalsify.org/vise.git/resource"
- "git.grassecon.net/urdt/ussd/internal/handlers"
- "git.grassecon.net/urdt/ussd/internal/storage"
-)
-
-var (
- logg = logging.NewVanilla()
- scriptDir = path.Join("services", "registration")
-)
-
-func main() {
- var dbDir string
- var size uint
- var sessionId string
- var engineDebug bool
- flag.StringVar(&sessionId, "session-id", "075xx2123", "session id")
- flag.StringVar(&dbDir, "dbdir", ".state", "database dir to read from")
- flag.BoolVar(&engineDebug, "d", false, "use engine debug output")
- flag.UintVar(&size, "s", 160, "max size of output")
- flag.Parse()
-
- logg.Infof("start command", "dbdir", dbDir, "outputsize", size)
-
- ctx := context.Background()
- ctx = context.WithValue(ctx, "SessionId", sessionId)
- pfp := path.Join(scriptDir, "pp.csv")
-
- cfg := engine.Config{
- Root: "root",
- SessionId: sessionId,
- OutputSize: uint32(size),
- FlagCount: uint32(20),
- }
-
- resourceDir := scriptDir
- menuStorageService := storage.NewMenuStorageService(dbDir, resourceDir)
-
- err := menuStorageService.EnsureDbDir()
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- rs, err := menuStorageService.GetResource(ctx)
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- pe, err := menuStorageService.GetPersister(ctx)
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- userdatastore, err := menuStorageService.GetUserdataDb(ctx)
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- dbResource, ok := rs.(*resource.DbResource)
- if !ok {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- lhs, err := handlers.NewLocalHandlerService(pfp, true, dbResource, cfg, rs)
- lhs.SetDataStore(&userdatastore)
- lhs.SetPersister(pe)
-
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- hl, err := lhs.GetHandler()
- if err != nil {
- fmt.Fprintf(os.Stderr, err.Error())
- os.Exit(1)
- }
-
- en := lhs.GetEngine()
- en = en.WithFirst(hl.Init)
- if engineDebug {
- en = en.WithDebug(nil)
- }
-
- err = engine.Loop(ctx, en, os.Stdin, os.Stdout, nil)
- if err != nil {
- fmt.Fprintf(os.Stderr, "loop exited with error: %v\n", err)
- os.Exit(1)
- }
-}
-
-
-
package handlers
-
-import (
- "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/urdt/ussd/internal/handlers/ussd"
- "git.grassecon.net/urdt/ussd/internal/storage"
-)
-
-type BaseSessionHandler struct {
- cfgTemplate engine.Config
- rp RequestParser
- rs resource.Resource
- hn *ussd.Handlers
- provider storage.StorageProvider
-}
-
-func NewBaseSessionHandler(cfg engine.Config, rs resource.Resource, stateDb db.Db, userdataDb db.Db, rp RequestParser, hn *ussd.Handlers) *BaseSessionHandler {
- return &BaseSessionHandler{
- cfgTemplate: cfg,
- rs: rs,
- hn: hn,
- rp: rp,
- provider: storage.NewSimpleStorageProvider(stateDb, userdataDb),
- }
-}
-
-func(f* BaseSessionHandler) Shutdown() {
- err := f.provider.Close()
- if err != nil {
- logg.Errorf("handler shutdown error", "err", err)
- }
-}
-
-func(f *BaseSessionHandler) 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 *BaseSessionHandler) 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.Config.SessionId)
- if err != nil {
- logg.ErrorCtxf(rqs.Ctx, "", "storage get error", err)
- return rqs, ErrStorage
- }
-
- f.hn = f.hn.WithPersister(rqs.Storage.Persister)
- eni := f.GetEngine(rqs.Config, f.rs, rqs.Storage.Persister)
- en, ok := eni.(*engine.DefaultEngine)
- if !ok {
- perr := f.provider.Put(rqs.Config.SessionId, rqs.Storage)
- rqs.Storage = nil
- if perr != nil {
- logg.ErrorCtxf(rqs.Ctx, "", "storage put error", perr)
- }
- return rqs, 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.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 *BaseSessionHandler) Output(rqs RequestSession) (RequestSession, error) {
- var err error
- _, err = rqs.Engine.Flush(rqs.Ctx, rqs.Writer)
- return rqs, err
-}
-
-func(f *BaseSessionHandler) Reset(rqs RequestSession) (RequestSession, error) {
- defer f.provider.Put(rqs.Config.SessionId, rqs.Storage)
- return rqs, rqs.Engine.Finish()
-}
-
-func(f *BaseSessionHandler) GetConfig() engine.Config {
- return f.cfgTemplate
-}
-
-func(f *BaseSessionHandler) GetRequestParser() RequestParser {
- return f.rp
-}
-
-
-
package handlers
-
-import (
- "git.defalsify.org/vise.git/asm"
- "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/urdt/ussd/internal/handlers/ussd"
-)
-
-type HandlerService interface {
- GetHandler() (*ussd.Handlers, error)
-}
-
-func getParser(fp string, debug bool) (*asm.FlagParser, error) {
- flagParser := asm.NewFlagParser().WithDebug()
- _, err := flagParser.Load(fp)
- if err != nil {
- return nil, err
- }
- return flagParser, nil
-}
-
-type LocalHandlerService struct {
- Parser *asm.FlagParser
- DbRs *resource.DbResource
- Pe *persist.Persister
- UserdataStore *db.Db
- Cfg engine.Config
- Rs resource.Resource
-}
-
-func NewLocalHandlerService(fp string, debug bool, dbResource *resource.DbResource, cfg engine.Config, rs resource.Resource) (*LocalHandlerService, error) {
- parser, err := getParser(fp, debug)
- if err != nil {
- return nil, err
- }
- return &LocalHandlerService{
- Parser: parser,
- DbRs: dbResource,
- Cfg: cfg,
- Rs: rs,
- }, nil
-}
-
-func (ls *LocalHandlerService) SetPersister(Pe *persist.Persister) {
- ls.Pe = Pe
-}
-
-func (ls *LocalHandlerService) SetDataStore(db *db.Db) {
- ls.UserdataStore = db
-}
-
-func (ls *LocalHandlerService) GetHandler() (*ussd.Handlers, error) {
- ussdHandlers, err := ussd.NewHandlers(ls.Parser, *ls.UserdataStore)
- if err != nil {
- return nil, err
- }
- ussdHandlers = ussdHandlers.WithPersister(ls.Pe)
- ls.DbRs.AddLocalFunc("set_language", ussdHandlers.SetLanguage)
- ls.DbRs.AddLocalFunc("create_account", ussdHandlers.CreateAccount)
- ls.DbRs.AddLocalFunc("save_pin", ussdHandlers.SavePin)
- ls.DbRs.AddLocalFunc("verify_pin", ussdHandlers.VerifyPin)
- ls.DbRs.AddLocalFunc("check_identifier", ussdHandlers.CheckIdentifier)
- ls.DbRs.AddLocalFunc("check_account_status", ussdHandlers.CheckAccountStatus)
- ls.DbRs.AddLocalFunc("authorize_account", ussdHandlers.Authorize)
- ls.DbRs.AddLocalFunc("quit", ussdHandlers.Quit)
- ls.DbRs.AddLocalFunc("check_balance", ussdHandlers.CheckBalance)
- ls.DbRs.AddLocalFunc("validate_recipient", ussdHandlers.ValidateRecipient)
- ls.DbRs.AddLocalFunc("transaction_reset", ussdHandlers.TransactionReset)
- ls.DbRs.AddLocalFunc("max_amount", ussdHandlers.MaxAmount)
- ls.DbRs.AddLocalFunc("validate_amount", ussdHandlers.ValidateAmount)
- ls.DbRs.AddLocalFunc("reset_transaction_amount", ussdHandlers.ResetTransactionAmount)
- ls.DbRs.AddLocalFunc("get_recipient", ussdHandlers.GetRecipient)
- ls.DbRs.AddLocalFunc("get_sender", ussdHandlers.GetSender)
- ls.DbRs.AddLocalFunc("get_amount", ussdHandlers.GetAmount)
- ls.DbRs.AddLocalFunc("reset_incorrect", ussdHandlers.ResetIncorrectPin)
- ls.DbRs.AddLocalFunc("save_firstname", ussdHandlers.SaveFirstname)
- ls.DbRs.AddLocalFunc("save_familyname", ussdHandlers.SaveFamilyname)
- ls.DbRs.AddLocalFunc("save_gender", ussdHandlers.SaveGender)
- ls.DbRs.AddLocalFunc("save_location", ussdHandlers.SaveLocation)
- ls.DbRs.AddLocalFunc("save_yob", ussdHandlers.SaveYob)
- ls.DbRs.AddLocalFunc("save_offerings", ussdHandlers.SaveOfferings)
- ls.DbRs.AddLocalFunc("quit_with_balance", ussdHandlers.QuitWithBalance)
- ls.DbRs.AddLocalFunc("reset_account_authorized", ussdHandlers.ResetAccountAuthorized)
- ls.DbRs.AddLocalFunc("reset_allow_update", ussdHandlers.ResetAllowUpdate)
- ls.DbRs.AddLocalFunc("get_profile_info", ussdHandlers.GetProfileInfo)
- ls.DbRs.AddLocalFunc("verify_yob", ussdHandlers.VerifyYob)
- ls.DbRs.AddLocalFunc("reset_incorrect_date_format", ussdHandlers.ResetIncorrectYob)
- ls.DbRs.AddLocalFunc("initiate_transaction", ussdHandlers.InitiateTransaction)
- ls.DbRs.AddLocalFunc("save_temporary_pin", ussdHandlers.SaveTemporaryPin)
- ls.DbRs.AddLocalFunc("verify_new_pin", ussdHandlers.VerifyNewPin)
- ls.DbRs.AddLocalFunc("confirm_pin_change", ussdHandlers.ConfirmPinChange)
- ls.DbRs.AddLocalFunc("quit_with_help", ussdHandlers.QuitWithHelp)
- ls.DbRs.AddLocalFunc("fetch_custodial_balances", ussdHandlers.FetchCustodialBalances)
-
- return ussdHandlers, nil
-}
-
-// TODO: enable setting of sessionId on engine init time
-func (ls *LocalHandlerService) GetEngine() *engine.DefaultEngine {
- en := engine.NewEngine(ls.Cfg, ls.Rs)
- en = en.WithPersister(ls.Pe)
- return en
-}
-
-
-
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) (*models.BalanceResponse, error)
- CreateAccount() (*models.AccountResponse, error)
- CheckAccountStatus(trackingId string) (*models.TrackStatusResponse, 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) (*models.TrackStatusResponse, error) {
- resp, err := http.Get(config.TrackStatusURL + trackingId)
- if err != nil {
- return nil, err
- }
- defer resp.Body.Close()
-
- body, err := io.ReadAll(resp.Body)
- if err != nil {
- return nil, err
- }
-
- var trackResp models.TrackStatusResponse
- err = json.Unmarshal(body, &trackResp)
- if err != nil {
- return nil, err
- }
-
- // status := trackResp.Result.Transaction.Status
-
- return &trackResp, 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) (*models.BalanceResponse, error) {
-
- resp, err := http.Get(config.BalanceURL + publicKey)
- if err != nil {
- return nil, err
- }
- defer resp.Body.Close()
-
- body, err := io.ReadAll(resp.Body)
- if err != nil {
- return nil, err
- }
-
- var balanceResp models.BalanceResponse
- err = json.Unmarshal(body, &balanceResp)
- if err != nil {
- return nil, err
- }
-
- //balance := balanceResp.Result.Balance
- return &balanceResp, 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
-}
-
-
-
package ussd
-
-import (
- "bytes"
- "context"
- "fmt"
- "path"
- "regexp"
- "strconv"
- "strings"
-
- "git.defalsify.org/vise.git/asm"
-
- "git.defalsify.org/vise.git/cache"
- "git.defalsify.org/vise.git/db"
- "git.defalsify.org/vise.git/lang"
- "git.defalsify.org/vise.git/logging"
- "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/server"
- "git.grassecon.net/urdt/ussd/internal/utils"
- "gopkg.in/leonelquinteros/gotext.v1"
-)
-
-var (
- logg = logging.NewVanilla().WithDomain("ussdmenuhandler")
- scriptDir = path.Join("services", "registration")
- translationDir = path.Join(scriptDir, "locale")
-)
-
-// FlagManager handles centralized flag management
-type FlagManager struct {
- parser *asm.FlagParser
-}
-
-// NewFlagManager creates a new FlagManager instance
-func NewFlagManager(csvPath string) (*FlagManager, error) {
- parser := asm.NewFlagParser()
- _, err := parser.Load(csvPath)
- if err != nil {
- return nil, fmt.Errorf("failed to load flag parser: %v", err)
- }
-
- return &FlagManager{
- parser: parser,
- }, nil
-}
-
-// GetFlag retrieves a flag value by its label
-func (fm *FlagManager) GetFlag(label string) (uint32, error) {
- return fm.parser.GetFlag(label)
-}
-
-type Handlers struct {
- pe *persist.Persister
- st *state.State
- ca cache.Memory
- userdataStore utils.DataStore
- flagManager *asm.FlagParser
- accountService server.AccountServiceInterface
-}
-
-func NewHandlers(appFlags *asm.FlagParser, userdataStore db.Db) (*Handlers, error) {
- if userdataStore == nil {
- return nil, fmt.Errorf("cannot create handler with nil userdata store")
- }
- userDb := &utils.UserDataStore{
- Db: userdataStore,
- }
- h := &Handlers{
- userdataStore: userDb,
- flagManager: appFlags,
- accountService: &server.AccountService{},
- }
- return h, 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) WithPersister(pe *persist.Persister) *Handlers {
- if h.pe != nil {
- panic("persister already set")
- }
- h.pe = pe
- return h
-}
-
-func (h *Handlers) Init(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var r resource.Result
-
- if h.pe == nil {
- logg.WarnCtxf(ctx, "handler init called before it is ready or more than once", "state", h.st, "cache", h.ca)
- return r, nil
- }
- h.st = h.pe.GetState()
- h.ca = h.pe.GetMemory()
- if h.st == nil || h.ca == nil {
- logg.ErrorCtxf(ctx, "perister fail in handler", "state", h.st, "cache", h.ca)
- return r, fmt.Errorf("cannot get state and memory for handler")
- }
- h.pe = nil
-
- logg.DebugCtxf(ctx, "handler has been initialized", "state", h.st, "cache", h.ca)
-
- return r, nil
-}
-
-// SetLanguage sets the language across the menu
-func (h *Handlers) SetLanguage(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
-
- symbol, _ := h.st.Where()
- code := strings.Split(symbol, "_")[1]
-
- if !utils.IsValidISO639(code) {
- return res, nil
- }
- res.FlagSet = append(res.FlagSet, state.FLAG_LANG)
- res.Content = code
-
- languageSetFlag, err := h.flagManager.GetFlag("flag_language_set")
- if err != nil {
- return res, err
- }
- res.FlagSet = append(res.FlagSet, languageSetFlag)
-
- return res, nil
-}
-
-func (h *Handlers) createAccountNoExist(ctx context.Context, sessionId string, res *resource.Result) error {
- accountResp, err := h.accountService.CreateAccount()
- data := map[utils.DataTyp]string{
- utils.DATA_TRACKING_ID: accountResp.Result.TrackingId,
- utils.DATA_PUBLIC_KEY: accountResp.Result.PublicKey,
- utils.DATA_CUSTODIAL_ID: accountResp.Result.CustodialId.String(),
- }
-
- for key, value := range data {
- store := h.userdataStore
- err := store.WriteEntry(ctx, sessionId, key, []byte(value))
- if err != nil {
- return err
- }
- }
- flag_account_created, _ := h.flagManager.GetFlag("flag_account_created")
- res.FlagSet = append(res.FlagSet, flag_account_created)
- return err
-
-}
-
-// 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) {
- var res resource.Result
- var err error
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
- store := h.userdataStore
- _, err = store.ReadEntry(ctx, sessionId, utils.DATA_ACCOUNT_CREATED)
- if err != nil {
- if db.IsNotFound(err) {
- logg.Printf(logging.LVL_INFO, "Creating an account because it doesn't exist")
- err = h.createAccountNoExist(ctx, sessionId, &res)
- if err != nil {
- return res, err
- }
- }
- }
- 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) {
- var res resource.Result
- var err error
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- flag_incorrect_pin, _ := h.flagManager.GetFlag("flag_incorrect_pin")
-
- accountPIN := string(input)
- // Validate that the PIN is a 4-digit number
- if !isValidPIN(accountPIN) {
- res.FlagSet = append(res.FlagSet, flag_incorrect_pin)
- return res, nil
- }
-
- res.FlagReset = append(res.FlagReset, flag_incorrect_pin)
- store := h.userdataStore
- err = store.WriteEntry(ctx, sessionId, utils.DATA_ACCOUNT_PIN, []byte(accountPIN))
- if err != nil {
- return res, err
- }
- return res, nil
-}
-
-func (h *Handlers) VerifyNewPin(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- res := resource.Result{}
- _, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
- flag_valid_pin, _ := h.flagManager.GetFlag("flag_valid_pin")
- pinInput := string(input)
- // Validate that the PIN is a 4-digit number
- if isValidPIN(pinInput) {
- res.FlagSet = append(res.FlagSet, flag_valid_pin)
- } else {
- res.FlagReset = append(res.FlagReset, flag_valid_pin)
- }
-
- return res, nil
-}
-
-func (h *Handlers) SaveTemporaryPin(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
- var err error
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
- flag_incorrect_pin, _ := h.flagManager.GetFlag("flag_incorrect_pin")
-
- accountPIN := string(input)
-
- // Validate that the PIN is a 4-digit number
- if !isValidPIN(accountPIN) {
- res.FlagSet = append(res.FlagSet, flag_incorrect_pin)
- return res, nil
- }
- store := h.userdataStore
- err = store.WriteEntry(ctx, sessionId, utils.DATA_TEMPORARY_PIN, []byte(accountPIN))
- if err != nil {
- return res, err
- }
- return res, nil
-}
-
-func (h *Handlers) ConfirmPinChange(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
- flag_pin_mismatch, _ := h.flagManager.GetFlag("flag_pin_mismatch")
-
- store := h.userdataStore
- temporaryPin, err := store.ReadEntry(ctx, sessionId, utils.DATA_TEMPORARY_PIN)
- if err != nil {
- return res, err
- }
- if bytes.Equal(temporaryPin, input) {
- res.FlagReset = append(res.FlagReset, flag_pin_mismatch)
- } else {
- res.FlagSet = append(res.FlagSet, flag_pin_mismatch)
- }
- err = store.WriteEntry(ctx, sessionId, utils.DATA_ACCOUNT_PIN, []byte(temporaryPin))
- if err != nil {
- return res, err
- }
- 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) {
- var res resource.Result
-
- flag_valid_pin, _ := h.flagManager.GetFlag("flag_valid_pin")
- flag_pin_mismatch, _ := h.flagManager.GetFlag("flag_pin_mismatch")
- flag_pin_set, _ := h.flagManager.GetFlag("flag_pin_set")
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
- store := h.userdataStore
- AccountPin, err := store.ReadEntry(ctx, sessionId, utils.DATA_ACCOUNT_PIN)
- if err != nil {
- return res, err
- }
-
- if bytes.Equal(input, AccountPin) {
- res.FlagSet = []uint32{flag_valid_pin}
- res.FlagReset = []uint32{flag_pin_mismatch}
- res.FlagSet = append(res.FlagSet, flag_pin_set)
- } else {
- res.FlagSet = []uint32{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
- if ctx.Value("Language") != nil {
- lang := ctx.Value("Language").(lang.Language)
- code = lang.Code
- }
- return code
-}
-
-// SaveFirstname updates the first name in the gdbm with the provided input.
-func (h *Handlers) SaveFirstname(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
- var err error
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
- if len(input) > 0 {
- firstName := string(input)
- store := h.userdataStore
- err = store.WriteEntry(ctx, sessionId, utils.DATA_FIRST_NAME, []byte(firstName))
- if err != nil {
- return res, err
- }
- }
-
- return res, nil
-}
-
-// SaveFamilyname updates the family name in the gdbm with the provided input.
-func (h *Handlers) SaveFamilyname(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
- var err error
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- if len(input) > 0 {
- familyName := string(input)
- store := h.userdataStore
- err = store.WriteEntry(ctx, sessionId, utils.DATA_FAMILY_NAME, []byte(familyName))
- if err != nil {
- return res, err
- }
- } else {
- return res, fmt.Errorf("a family name cannot be less than one character")
- }
-
- return res, nil
-}
-
-// SaveYOB updates the Year of Birth(YOB) in the gdbm with the provided input.
-func (h *Handlers) SaveYob(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
- var err error
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- if len(input) == 4 {
- yob := string(input)
- store := h.userdataStore
- err = store.WriteEntry(ctx, sessionId, utils.DATA_YOB, []byte(yob))
- if err != nil {
- return res, err
- }
- }
-
- return res, nil
-}
-
-// SaveLocation updates the location in the gdbm with the provided input.
-func (h *Handlers) SaveLocation(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
- var err error
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- if len(input) > 0 {
- location := string(input)
- store := h.userdataStore
- err = store.WriteEntry(ctx, sessionId, utils.DATA_LOCATION, []byte(location))
- if err != nil {
- return res, err
- }
- }
-
- return res, nil
-}
-
-// SaveGender updates the gender in the gdbm with the provided input.
-func (h *Handlers) SaveGender(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- symbol, _ := h.st.Where()
- var res resource.Result
- var err error
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- gender := strings.Split(symbol, "_")[1]
- store := h.userdataStore
- err = store.WriteEntry(ctx, sessionId, utils.DATA_GENDER, []byte(gender))
- if err != nil {
- return res, nil
- }
-
- return res, nil
-}
-
-// SaveOfferings updates the offerings(goods and services provided by the user) in the gdbm with the provided input.
-func (h *Handlers) SaveOfferings(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
- var err error
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- if len(input) > 0 {
- offerings := string(input)
- store := h.userdataStore
- err = store.WriteEntry(ctx, sessionId, utils.DATA_OFFERINGS, []byte(offerings))
- if err != nil {
- return res, nil
- }
- }
-
- 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) {
- var res resource.Result
-
- flag_allow_update, _ := h.flagManager.GetFlag("flag_allow_update")
-
- res.FlagReset = append(res.FlagReset, 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) {
- var res resource.Result
-
- flag_account_authorized, _ := h.flagManager.GetFlag("flag_account_authorized")
-
- res.FlagReset = append(res.FlagReset, 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) {
- var res resource.Result
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- store := h.userdataStore
- publicKey, _ := store.ReadEntry(ctx, sessionId, utils.DATA_PUBLIC_KEY)
-
- 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) {
- var res resource.Result
- var err error
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- flag_incorrect_pin, _ := h.flagManager.GetFlag("flag_incorrect_pin")
- flag_account_authorized, _ := h.flagManager.GetFlag("flag_account_authorized")
- flag_allow_update, _ := h.flagManager.GetFlag("flag_allow_update")
-
- store := h.userdataStore
- AccountPin, err := store.ReadEntry(ctx, sessionId, utils.DATA_ACCOUNT_PIN)
- if err != nil {
- return res, err
- }
- if len(input) == 4 {
- if bytes.Equal(input, AccountPin) {
- if h.st.MatchFlag(flag_account_authorized, false) {
- res.FlagReset = append(res.FlagReset, flag_incorrect_pin)
- res.FlagSet = append(res.FlagSet, flag_allow_update, flag_account_authorized)
- } else {
- res.FlagSet = append(res.FlagSet, flag_allow_update)
- res.FlagReset = append(res.FlagReset, flag_account_authorized)
- }
- } else {
- res.FlagSet = append(res.FlagSet, flag_incorrect_pin)
- res.FlagReset = append(res.FlagReset, flag_account_authorized)
- return res, nil
- }
- } else {
- return res, nil
- }
- 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) {
- var res resource.Result
- flag_incorrect_pin, _ := h.flagManager.GetFlag("flag_incorrect_pin")
- res.FlagReset = append(res.FlagReset, 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) {
- var res resource.Result
-
- flag_account_success, _ := h.flagManager.GetFlag("flag_account_success")
- flag_account_pending, _ := h.flagManager.GetFlag("flag_account_pending")
- flag_api_error, _ := h.flagManager.GetFlag("flag_api_call_error")
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
- store := h.userdataStore
- trackingId, err := store.ReadEntry(ctx, sessionId, utils.DATA_TRACKING_ID)
- if err != nil {
- return res, err
- }
-
- accountStatus, err := h.accountService.CheckAccountStatus(string(trackingId))
- if err != nil {
- fmt.Println("Error checking account status:", err)
- return res, err
- }
- if !accountStatus.Ok {
- res.FlagSet = append(res.FlagSet, flag_api_error)
- return res, err
- }
- res.FlagReset = append(res.FlagReset, flag_api_error)
- status := accountStatus.Result.Transaction.Status
-
- err = store.WriteEntry(ctx, sessionId, utils.DATA_ACCOUNT_STATUS, []byte(status))
- if err != nil {
- return res, nil
- }
- if accountStatus.Result.Transaction.Status == "SUCCESS" {
- res.FlagSet = append(res.FlagSet, flag_account_success)
- res.FlagReset = append(res.FlagReset, flag_account_pending)
- } else {
- res.FlagReset = append(res.FlagReset, flag_account_success)
- res.FlagSet = append(res.FlagSet, 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) {
- var res resource.Result
-
- flag_account_authorized, _ := h.flagManager.GetFlag("flag_account_authorized")
-
- 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, flag_account_authorized)
- return res, nil
-}
-
-// QuitWithHelp displays helpline information then exits the menu
-func (h *Handlers) QuitWithHelp(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
-
- flag_account_authorized, _ := h.flagManager.GetFlag("flag_account_authorized")
-
- code := codeFromCtx(ctx)
- l := gotext.NewLocale(translationDir, code)
- l.AddDomain("default")
-
- res.Content = l.Get("For more help,please call: 0757628885")
- res.FlagReset = append(res.FlagReset, 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) {
- var res resource.Result
- var err error
-
- flag_incorrect_date_format, _ := h.flagManager.GetFlag("flag_incorrect_date_format")
-
- date := string(input)
- _, err = strconv.Atoi(date)
- if err != nil {
- // If conversion fails, input is not numeric
- res.FlagSet = append(res.FlagSet, flag_incorrect_date_format)
- return res, nil
- }
-
- if len(date) == 4 {
- res.FlagReset = append(res.FlagReset, flag_incorrect_date_format)
- } else {
- res.FlagSet = append(res.FlagSet, 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) {
- var res resource.Result
-
- flag_incorrect_date_format, _ := h.flagManager.GetFlag("flag_incorrect_date_format")
-
- res.FlagReset = append(res.FlagReset, 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) {
- var res resource.Result
- var err error
-
- flag_api_error, _ := h.flagManager.GetFlag("flag_api_call_error")
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- store := h.userdataStore
- publicKey, err := store.ReadEntry(ctx, sessionId, utils.DATA_PUBLIC_KEY)
- if err != nil {
- return res, err
- }
-
- balanceResponse, err := h.accountService.CheckBalance(string(publicKey))
- if err != nil {
- return res, nil
- }
- if !balanceResponse.Ok {
- res.FlagSet = append(res.FlagSet, flag_api_error)
- return res, nil
- }
- res.FlagReset = append(res.FlagReset, flag_api_error)
- balance := balanceResponse.Result.Balance
- res.Content = balance
-
- return res, nil
-}
-
-func (h *Handlers) FetchCustodialBalances(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
- flag_api_error, _ := h.flagManager.GetFlag("flag_api_call_error")
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
- symbol, _ := h.st.Where()
- balanceType := strings.Split(symbol, "_")[0]
-
- store := h.userdataStore
- publicKey, err := store.ReadEntry(ctx, sessionId, utils.DATA_PUBLIC_KEY)
- if err != nil {
- return res, err
- }
-
- balanceResponse, err := h.accountService.CheckBalance(string(publicKey))
- if err != nil {
- return res, nil
- }
- if !balanceResponse.Ok {
- res.FlagSet = append(res.FlagSet, flag_api_error)
- return res, nil
- }
- res.FlagReset = append(res.FlagReset, flag_api_error)
- balance := balanceResponse.Result.Balance
-
- switch balanceType {
- case "my":
- res.Content = fmt.Sprintf("Your balance is %s", balance)
- case "community":
- res.Content = fmt.Sprintf("Your community balance is %s", balance)
- default:
- break
- }
- 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) {
- var res resource.Result
- var err error
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- recipient := string(input)
-
- flag_invalid_recipient, _ := h.flagManager.GetFlag("flag_invalid_recipient")
-
- if recipient != "0" {
- // mimic invalid number check
- if recipient == "000" {
- res.FlagSet = append(res.FlagSet, flag_invalid_recipient)
- res.Content = recipient
-
- return res, nil
- }
- store := h.userdataStore
- err = store.WriteEntry(ctx, sessionId, utils.DATA_RECIPIENT, []byte(recipient))
- if err != nil {
- return res, nil
- }
- }
-
- 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) {
- var res resource.Result
- var err error
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- flag_invalid_recipient, _ := h.flagManager.GetFlag("flag_invalid_recipient")
- flag_invalid_recipient_with_invite, _ := h.flagManager.GetFlag("flag_invalid_recipient_with_invite")
- store := h.userdataStore
- err = store.WriteEntry(ctx, sessionId, utils.DATA_AMOUNT, []byte(""))
- if err != nil {
- return res, nil
- }
-
- err = store.WriteEntry(ctx, sessionId, utils.DATA_RECIPIENT, []byte(""))
- if err != nil {
- return res, nil
- }
-
- res.FlagReset = append(res.FlagReset, flag_invalid_recipient, 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) {
- var res resource.Result
- var err error
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- flag_invalid_amount, _ := h.flagManager.GetFlag("flag_invalid_amount")
- store := h.userdataStore
- err = store.WriteEntry(ctx, sessionId, utils.DATA_AMOUNT, []byte(""))
- if err != nil {
- return res, nil
- }
-
- res.FlagReset = append(res.FlagReset, 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) {
- var res resource.Result
- var err error
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
- store := h.userdataStore
- publicKey, _ := store.ReadEntry(ctx, sessionId, utils.DATA_PUBLIC_KEY)
-
- balanceResp, err := h.accountService.CheckBalance(string(publicKey))
- if err != nil {
- return res, nil
- }
- balance := balanceResp.Result.Balance
-
- 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) {
- var res resource.Result
- var err error
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- flag_invalid_amount, _ := h.flagManager.GetFlag("flag_invalid_amount")
-
- store := h.userdataStore
- publicKey, _ := store.ReadEntry(ctx, sessionId, utils.DATA_PUBLIC_KEY)
-
- amountStr := string(input)
-
- balanceRes, err := h.accountService.CheckBalance(string(publicKey))
- balanceStr := balanceRes.Result.Balance
-
- 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, flag_invalid_amount)
- res.Content = amountStr
- return res, nil
- }
-
- inputAmount, err := strconv.ParseFloat(matches[1], 64)
- if err != nil {
- res.FlagSet = append(res.FlagSet, flag_invalid_amount)
- res.Content = amountStr
- return res, nil
- }
-
- if inputAmount > balanceValue {
- res.FlagSet = append(res.FlagSet, flag_invalid_amount)
- res.Content = amountStr
- return res, nil
- }
-
- res.Content = fmt.Sprintf("%.3f", inputAmount) // Format to 3 decimal places
- err = store.WriteEntry(ctx, sessionId, utils.DATA_AMOUNT, []byte(amountStr))
- if err != nil {
- return res, err
- }
-
- return res, nil
-}
-
-// GetRecipient returns the transaction recipient from the gdbm.
-func (h *Handlers) GetRecipient(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
- store := h.userdataStore
- recipient, _ := store.ReadEntry(ctx, sessionId, utils.DATA_RECIPIENT)
-
- res.Content = string(recipient)
-
- return res, nil
-}
-
-// GetSender retrieves the public key from the Gdbm Db
-func (h *Handlers) GetSender(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- store := h.userdataStore
- publicKey, _ := store.ReadEntry(ctx, sessionId, utils.DATA_PUBLIC_KEY)
-
- res.Content = string(publicKey)
-
- return res, nil
-}
-
-// GetAmount retrieves the amount from teh Gdbm Db
-func (h *Handlers) GetAmount(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
-
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
- store := h.userdataStore
- amount, _ := store.ReadEntry(ctx, sessionId, utils.DATA_AMOUNT)
-
- 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) {
- var res resource.Result
- var err error
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- flag_account_authorized, _ := h.flagManager.GetFlag("flag_account_authorized")
-
- code := codeFromCtx(ctx)
- l := gotext.NewLocale(translationDir, code)
- l.AddDomain("default")
-
- store := h.userdataStore
- publicKey, err := store.ReadEntry(ctx, sessionId, utils.DATA_PUBLIC_KEY)
- 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, flag_account_authorized)
- return res, nil
-}
-
-// InitiateTransaction returns a confirmation and resets the transaction data
-// on the gdbm store.
-func (h *Handlers) InitiateTransaction(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
- var err error
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
-
- 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
- store := h.userdataStore
- publicKey, _ := store.ReadEntry(ctx, sessionId, utils.DATA_PUBLIC_KEY)
-
- amount, _ := store.ReadEntry(ctx, sessionId, utils.DATA_AMOUNT)
-
- recipient, _ := store.ReadEntry(ctx, sessionId, utils.DATA_RECIPIENT)
-
- 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.flagManager.GetFlag("flag_account_authorized")
- if err != nil {
- return res, err
- }
-
- res.FlagReset = append(res.FlagReset, account_authorized_flag)
- return res, nil
-}
-
-func (h *Handlers) GetProfileInfo(ctx context.Context, sym string, input []byte) (resource.Result, error) {
- var res resource.Result
- var defaultValue string
- sessionId, ok := ctx.Value("SessionId").(string)
- if !ok {
- return res, fmt.Errorf("missing session")
- }
- language, ok := ctx.Value("Language").(lang.Language)
- if !ok {
- return res, fmt.Errorf("value for 'Language' is not of type lang.Language")
- }
- code := language.Code
- if code == "swa" {
- defaultValue = "Haipo"
- } else {
- defaultValue = "Not Provided"
- }
-
- // Helper function to handle nil byte slices and convert them to string
- getEntryOrDefault := func(entry []byte, err error) string {
- if err != nil || entry == nil {
- return defaultValue
- }
- return string(entry)
- }
- store := h.userdataStore
- // Retrieve user data as strings with fallback to defaultValue
- firstName := getEntryOrDefault(store.ReadEntry(ctx, sessionId, utils.DATA_FIRST_NAME))
- familyName := getEntryOrDefault(store.ReadEntry(ctx, sessionId, utils.DATA_FAMILY_NAME))
- yob := getEntryOrDefault(store.ReadEntry(ctx, sessionId, utils.DATA_YOB))
- gender := getEntryOrDefault(store.ReadEntry(ctx, sessionId, utils.DATA_GENDER))
- location := getEntryOrDefault(store.ReadEntry(ctx, sessionId, utils.DATA_LOCATION))
- offerings := getEntryOrDefault(store.ReadEntry(ctx, sessionId, utils.DATA_OFFERINGS))
-
- // Construct the full name
- name := defaultValue
- if familyName != defaultValue {
- if firstName == defaultValue {
- name = familyName
- } else {
- name = firstName + " " + familyName
- }
- }
-
- // Calculate age from year of birth
- age := defaultValue
- if yob != defaultValue {
- if yobInt, err := strconv.Atoi(yob); err == nil {
- age = strconv.Itoa(utils.CalculateAgeWithYOB(yobInt))
- } else {
- return res, fmt.Errorf("invalid year of birth: %v", err)
- }
- }
- switch language.Code {
- case "eng":
- res.Content = fmt.Sprintf(
- "Name: %s\nGender: %s\nAge: %s\nLocation: %s\nYou provide: %s\n",
- name, gender, age, location, offerings,
- )
- case "swa":
- res.Content = fmt.Sprintf(
- "Jina: %s\nJinsia: %s\nUmri: %s\nEneo: %s\nUnauza: %s\n",
- name, gender, age, location, offerings,
- )
- default:
- res.Content = fmt.Sprintf(
- "Name: %s\nGender: %s\nAge: %s\nLocation: %s\nYou provide: %s\n",
- name, gender, age, location, offerings,
- )
- }
-
- return res, nil
-}
-
-
-
package http
-
-import (
- "io"
- "net/http"
-
- "git.grassecon.net/urdt/ussd/internal/handlers"
-)
-
-type ATSessionHandler struct {
- *SessionHandler
-}
-
-func NewATSessionHandler(h handlers.RequestHandler) *ATSessionHandler {
- return &ATSessionHandler{
- SessionHandler: ToSessionHandler(h),
- }
-}
-
-func (ash *ATSessionHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) {
- var code int
- var err error
-
- rqs := handlers.RequestSession{
- Ctx: req.Context(),
- Writer: w,
- }
-
- rp := ash.GetRequestParser()
- cfg := ash.GetConfig()
- cfg.SessionId, err = rp.GetSessionId(req)
- if err != nil {
- logg.ErrorCtxf(rqs.Ctx, "", "header processing error", err)
- ash.writeError(w, 400, err)
- return
- }
- rqs.Config = cfg
- rqs.Input, err = rp.GetInput(req)
- if err != nil {
- logg.ErrorCtxf(rqs.Ctx, "", "header processing error", err)
- ash.writeError(w, 400, err)
- return
- }
-
- rqs, err = ash.Process(rqs)
- switch err {
- case nil: // set code to 200 if no err
- code = 200
- case handlers.ErrStorage, handlers.ErrEngineInit, handlers.ErrEngineExec, handlers.ErrEngineType:
- code = 500
- default:
- code = 500
- }
-
- if code != 200 {
- ash.writeError(w, 500, err)
- return
- }
-
- w.WriteHeader(200)
- w.Header().Set("Content-Type", "text/plain")
- rqs, err = ash.Output(rqs)
- if err != nil {
- ash.writeError(w, 500, err)
- return
- }
-
- rqs, err = ash.Reset(rqs)
- if err != nil {
- ash.writeError(w, 500, err)
- return
- }
-}
-
-func (ash *ATSessionHandler) Output(rqs handlers.RequestSession) (handlers.RequestSession, error) {
- var err error
- var prefix string
-
- if rqs.Continue {
- prefix = "CON "
- } else {
- prefix = "END "
- }
-
- _, err = io.WriteString(rqs.Writer, prefix)
- if err != nil {
- return rqs, err
- }
-
- _, err = rqs.Engine.Flush(rqs.Ctx, rqs.Writer)
- return rqs, err
-}
-
-
package http
-
-import (
- "io/ioutil"
- "net/http"
- "strconv"
-
- "git.defalsify.org/vise.git/logging"
-
- "git.grassecon.net/urdt/ussd/internal/handlers"
-)
-
-var (
- logg = logging.NewVanilla().WithDomain("httpserver")
-)
-
-type DefaultRequestParser struct {
-}
-
-
-func(rp *DefaultRequestParser) GetSessionId(rq any) (string, error) {
- rqv, ok := rq.(*http.Request)
- if !ok {
- return "", handlers.ErrInvalidRequest
- }
- v := rqv.Header.Get("X-Vise-Session")
- if v == "" {
- return "", handlers.ErrSessionMissing
- }
- return v, nil
-}
-
-func(rp *DefaultRequestParser) GetInput(rq any) ([]byte, error) {
- rqv, ok := rq.(*http.Request)
- if !ok {
- return nil, handlers.ErrInvalidRequest
- }
- defer rqv.Body.Close()
- v, err := ioutil.ReadAll(rqv.Body)
- if err != nil {
- return nil, err
- }
- return v, nil
-}
-
-type SessionHandler struct {
- handlers.RequestHandler
-}
-
-func ToSessionHandler(h handlers.RequestHandler) *SessionHandler {
- return &SessionHandler{
- RequestHandler: h,
- }
-}
-
-func(f *SessionHandler) 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{})
- if err != nil {
- logg.Errorf("error writing error!!", "err", err, "olderr", s)
- w.WriteHeader(500)
- }
- return
-}
-
-func(f *SessionHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) {
- var code int
- var err error
- var perr error
-
- rqs := handlers.RequestSession{
- Ctx: req.Context(),
- Writer: w,
- }
-
- rp := f.GetRequestParser()
- cfg := f.GetConfig()
- cfg.SessionId, err = rp.GetSessionId(req)
- if err != nil {
- logg.ErrorCtxf(rqs.Ctx, "", "header processing error", err)
- f.writeError(w, 400, err)
- }
- rqs.Config = cfg
- rqs.Input, err = rp.GetInput(req)
- if err != nil {
- logg.ErrorCtxf(rqs.Ctx, "", "header processing error", err)
- f.writeError(w, 400, err)
- return
- }
-
- rqs, err = f.Process(rqs)
- switch err {
- case handlers.ErrStorage:
- code = 500
- case handlers.ErrEngineInit:
- code = 500
- case handlers.ErrEngineExec:
- code = 500
- default:
- code = 200
- }
-
- if code != 200 {
- f.writeError(w, 500, err)
- return
- }
-
- w.WriteHeader(200)
- w.Header().Set("Content-Type", "text/plain")
- rqs, err = f.Output(rqs)
- rqs, perr = f.Reset(rqs)
- if err != nil {
- f.writeError(w, 500, err)
- return
- }
- if perr != nil {
- f.writeError(w, 500, perr)
- return
- }
-}
-
-
-
package mocks
-
-import (
- "context"
-
- "git.defalsify.org/vise.git/lang"
- "github.com/stretchr/testify/mock"
-)
-
-type MockDb struct {
- mock.Mock
-}
-
-func (m *MockDb) SetPrefix(prefix uint8) {
- m.Called(prefix)
-}
-
-func (m *MockDb) Prefix() uint8 {
- args := m.Called()
- return args.Get(0).(uint8)
-}
-
-func (m *MockDb) Safe() bool {
- args := m.Called()
- return args.Get(0).(bool)
-}
-
-func (m *MockDb) SetLanguage(language *lang.Language) {
- m.Called(language)
-}
-
-func (m *MockDb) SetLock(uint8, bool) error {
- args := m.Called()
- return args.Error(0)
-}
-
-func (m *MockDb) Connect(ctx context.Context, connectionStr string) error {
- args := m.Called(ctx, connectionStr)
- return args.Error(0)
-}
-
-func (m *MockDb) SetSession(sessionId string) {
- m.Called(sessionId)
-}
-
-func (m *MockDb) Put(ctx context.Context, key, value []byte) error {
- args := m.Called(ctx, key, value)
- return args.Error(0)
-}
-
-func (m *MockDb) Get(ctx context.Context, key []byte) ([]byte, error) {
- args := m.Called(ctx, key)
- return nil, args.Error(0)
-}
-
-func (m *MockDb) Close() error {
- args := m.Called(nil)
- return args.Error(0)
-}
-
-
-
package httpmocks
-
-import (
- "context"
- "io"
-)
-
-// MockEngine implements the engine.Engine interface for testing
-type MockEngine struct {
- InitFunc func(context.Context) (bool, error)
- ExecFunc func(context.Context, []byte) (bool, error)
- FlushFunc func(context.Context, io.Writer) (int, error)
- FinishFunc func() error
-}
-
-func (m *MockEngine) Init(ctx context.Context) (bool, error) {
- return m.InitFunc(ctx)
-}
-
-func (m *MockEngine) Exec(ctx context.Context, input []byte) (bool, error) {
- return m.ExecFunc(ctx, input)
-}
-
-func (m *MockEngine) Flush(ctx context.Context, w io.Writer) (int, error) {
- return m.FlushFunc(ctx, w)
-}
-
-func (m *MockEngine) Finish() error {
- return m.FinishFunc()
-}
-
-
-
package httpmocks
-
-import (
- "git.defalsify.org/vise.git/engine"
- "git.defalsify.org/vise.git/persist"
- "git.defalsify.org/vise.git/resource"
- "git.grassecon.net/urdt/ussd/internal/handlers"
-)
-
-// MockRequestHandler implements handlers.RequestHandler interface for testing
-type MockRequestHandler struct {
- ProcessFunc func(handlers.RequestSession) (handlers.RequestSession, error)
- GetConfigFunc func() engine.Config
- GetEngineFunc func(cfg engine.Config, rs resource.Resource, pe *persist.Persister) engine.Engine
- OutputFunc func(rs handlers.RequestSession) (handlers.RequestSession, error)
- ResetFunc func(rs handlers.RequestSession) (handlers.RequestSession, error)
- ShutdownFunc func()
- GetRequestParserFunc func() handlers.RequestParser
-}
-
-func (m *MockRequestHandler) Process(rqs handlers.RequestSession) (handlers.RequestSession, error) {
- return m.ProcessFunc(rqs)
-}
-
-func (m *MockRequestHandler) GetConfig() engine.Config {
- return m.GetConfigFunc()
-}
-
-func (m *MockRequestHandler) GetEngine(cfg engine.Config, rs resource.Resource, pe *persist.Persister) engine.Engine {
- return m.GetEngineFunc(cfg, rs, pe)
-}
-
-func (m *MockRequestHandler) Output(rs handlers.RequestSession) (handlers.RequestSession, error) {
- return m.OutputFunc(rs)
-}
-
-func (m *MockRequestHandler) Reset(rs handlers.RequestSession) (handlers.RequestSession, error) {
- return m.ResetFunc(rs)
-}
-
-func (m *MockRequestHandler) Shutdown() {
- m.ShutdownFunc()
-}
-
-func (m *MockRequestHandler) GetRequestParser() handlers.RequestParser {
- return m.GetRequestParserFunc()
-}
-
-
-
package httpmocks
-
-// MockRequestParser implements the handlers.RequestParser interface for testing
-type MockRequestParser struct {
- GetSessionIdFunc func(any) (string, error)
- GetInputFunc func(any) ([]byte, error)
-}
-
-func (m *MockRequestParser) GetSessionId(rq any) (string, error) {
- return m.GetSessionIdFunc(rq)
-}
-
-func (m *MockRequestParser) GetInput(rq any) ([]byte, error) {
- return m.GetInputFunc(rq)
-}
-
-
-
package httpmocks
-
-import "net/http"
-
-// MockWriter implements a mock io.Writer for testing
-type MockWriter struct {
- WriteStringCalled bool
- WrittenString string
-}
-
-func (m *MockWriter) Write(p []byte) (n int, err error) {
- return len(p), nil
-}
-
-func (m *MockWriter) WriteString(s string) (n int, err error) {
- m.WriteStringCalled = true
- m.WrittenString = s
- return len(s), nil
-}
-
-func (m *MockWriter) Header() http.Header {
- return http.Header{}
-}
-
-func (m *MockWriter) WriteHeader(statusCode int) {}
-
-
package mocks
-
-import (
- "git.grassecon.net/urdt/ussd/internal/models"
- "github.com/stretchr/testify/mock"
-)
-
-// MockAccountService implements AccountServiceInterface for testing
-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) CheckBalance(publicKey string) (*models.BalanceResponse, error) {
- args := m.Called(publicKey)
- return args.Get(0).(*models.BalanceResponse), args.Error(1)
-}
-
-func (m *MockAccountService) CheckAccountStatus(trackingId string) (*models.TrackStatusResponse, error) {
- args := m.Called(trackingId)
- return args.Get(0).(*models.TrackStatusResponse), args.Error(1)
-}
-
-
package mocks
-
-import (
- "context"
-
- "git.defalsify.org/vise.git/db"
- "git.grassecon.net/urdt/ussd/internal/utils"
- "github.com/stretchr/testify/mock"
-)
-
-type MockUserDataStore struct {
- db.Db
- mock.Mock
-}
-
-func (m *MockUserDataStore) ReadEntry(ctx context.Context, sessionId string, typ utils.DataTyp) ([]byte, error) {
- args := m.Called(ctx, sessionId, typ)
- return args.Get(0).([]byte), args.Error(1)
-}
-
-func (m *MockUserDataStore) WriteEntry(ctx context.Context, sessionId string, typ utils.DataTyp, value []byte) error {
- args := m.Called(ctx, sessionId, typ, value)
- return args.Error(0)
-}
-
-
-
package storage
-
-import (
- "context"
-
- "git.defalsify.org/vise.git/db"
- "git.defalsify.org/vise.git/lang"
- gdbmdb "git.defalsify.org/vise.git/db/gdbm"
-)
-
-var (
- dbC map[string]chan db.Db
-)
-
-type ThreadGdbmDb struct {
- db db.Db
- connStr string
-}
-
-func NewThreadGdbmDb() *ThreadGdbmDb {
- if dbC == nil {
- dbC = make(map[string]chan db.Db)
- }
- return &ThreadGdbmDb{}
-}
-
-func(tdb *ThreadGdbmDb) Connect(ctx context.Context, connStr string) error {
- var ok bool
- _, ok = dbC[connStr]
- if ok {
- logg.WarnCtxf(ctx, "already registered thread gdbm, skipping", "connStr", connStr)
- return nil
- }
- gdb := gdbmdb.NewGdbmDb()
- err := gdb.Connect(ctx, connStr)
- if err != nil {
- return err
- }
- dbC[connStr] = make(chan db.Db, 1)
- dbC[connStr]<- gdb
- tdb.connStr = connStr
- return nil
-}
-
-func(tdb *ThreadGdbmDb) reserve() {
- if tdb.db == nil {
- tdb.db = <-dbC[tdb.connStr]
- }
-}
-
-func(tdb *ThreadGdbmDb) release() {
- if tdb.db == nil {
- return
- }
- dbC[tdb.connStr] <- tdb.db
- tdb.db = nil
-}
-
-func(tdb *ThreadGdbmDb) SetPrefix(pfx uint8) {
- tdb.reserve()
- tdb.db.SetPrefix(pfx)
-}
-
-func(tdb *ThreadGdbmDb) SetSession(sessionId string) {
- tdb.reserve()
- tdb.db.SetSession(sessionId)
-}
-
-func(tdb *ThreadGdbmDb) SetLanguage(lng *lang.Language) {
- tdb.reserve()
- tdb.db.SetLanguage(lng)
-}
-
-func(tdb *ThreadGdbmDb) Safe() bool {
- tdb.reserve()
- v := tdb.db.Safe()
- tdb.release()
- return v
-}
-
-func(tdb *ThreadGdbmDb) Prefix() uint8 {
- tdb.reserve()
- v := tdb.db.Prefix()
- tdb.release()
- return v
-}
-
-func(tdb *ThreadGdbmDb) SetLock(typ uint8, locked bool) error {
- tdb.reserve()
- err := tdb.db.SetLock(typ, locked)
- tdb.release()
- return err
-}
-
-func(tdb *ThreadGdbmDb) Put(ctx context.Context, key []byte, val []byte) error {
- tdb.reserve()
- err := tdb.db.Put(ctx, key, val)
- tdb.release()
- return err
-}
-
-func(tdb *ThreadGdbmDb) Get(ctx context.Context, key []byte) ([]byte, error) {
- tdb.reserve()
- v, err := tdb.db.Get(ctx, key)
- tdb.release()
- return v, err
-}
-
-func(tdb *ThreadGdbmDb) Close() error {
- tdb.reserve()
- close(dbC[tdb.connStr])
- err := tdb.db.Close()
- tdb.db = nil
- return err
-}
-
-
-
package storage
-
-import (
- "git.defalsify.org/vise.git/db"
- "git.defalsify.org/vise.git/persist"
-)
-
-const (
- DATATYPE_CUSTOM = 128
-)
-
-type Storage struct {
- Persister *persist.Persister
- UserdataDb db.Db
-}
-
-type StorageProvider interface {
- Get(sessionId string) (*Storage, error)
- Put(sessionId string, storage *Storage) error
- Close() error
-}
-
-type SimpleStorageProvider struct {
- *Storage
-}
-
-func NewSimpleStorageProvider(stateStore db.Db, userdataStore db.Db) StorageProvider {
- pe := persist.NewPersister(stateStore)
- pe = pe.WithFlush()
- return &SimpleStorageProvider{
- Storage: &Storage{
- Persister: pe,
- UserdataDb: userdataStore,
- },
- }
-}
-
-func (p *SimpleStorageProvider) Get(sessionId string) (*Storage, error) {
- return p.Storage, nil
-}
-
-func (p *SimpleStorageProvider) Put(sessionId string, storage *Storage) error {
- return nil
-}
-
-func (p *SimpleStorageProvider) Close() error {
- return p.Storage.UserdataDb.Close()
-}
-
-
-
package storage
-
-import (
- "context"
- "fmt"
- "os"
- "path"
-
- "git.defalsify.org/vise.git/db"
- fsdb "git.defalsify.org/vise.git/db/fs"
- "git.defalsify.org/vise.git/persist"
- "git.defalsify.org/vise.git/resource"
- "git.defalsify.org/vise.git/logging"
-)
-
-var (
- logg = logging.NewVanilla().WithDomain("storage")
-)
-
-type StorageService interface {
- GetPersister(ctx context.Context) (*persist.Persister, error)
- GetUserdataDb(ctx context.Context) db.Db
- GetResource(ctx context.Context) (resource.Resource, error)
- EnsureDbDir() error
-}
-
-type MenuStorageService struct{
- dbDir string
- resourceDir string
- resourceStore db.Db
- stateStore db.Db
- userDataStore db.Db
-}
-
-func NewMenuStorageService(dbDir string, resourceDir string) *MenuStorageService {
- return &MenuStorageService{
- dbDir: dbDir,
- resourceDir: resourceDir,
- }
-}
-
-func (ms *MenuStorageService) GetPersister(ctx context.Context) (*persist.Persister, error) {
- ms.stateStore = NewThreadGdbmDb()
- storeFile := path.Join(ms.dbDir, "state.gdbm")
- err := ms.stateStore.Connect(ctx, storeFile)
- if err != nil {
- return nil, err
- }
- pr := persist.NewPersister(ms.stateStore)
- logg.TraceCtxf(ctx, "menu storage service", "persist", pr, "store", ms.stateStore)
- return pr, nil
-}
-
-func (ms *MenuStorageService) GetUserdataDb(ctx context.Context) (db.Db, error) {
- ms.userDataStore = NewThreadGdbmDb()
- storeFile := path.Join(ms.dbDir, "userdata.gdbm")
- err := ms.userDataStore.Connect(ctx, storeFile)
- if err != nil {
- return nil, err
- }
- return ms.userDataStore, nil
-}
-
-func (ms *MenuStorageService) GetResource(ctx context.Context) (resource.Resource, error) {
- ms.resourceStore = fsdb.NewFsDb()
- err := ms.resourceStore.Connect(ctx, ms.resourceDir)
- if err != nil {
- return nil, err
- }
- rfs := resource.NewDbResource(ms.resourceStore)
- return rfs, nil
-}
-
-func (ms *MenuStorageService) GetStateStore(ctx context.Context) (db.Db, error) {
- if ms.stateStore != nil {
- panic("set up store when already exists")
- }
- ms.stateStore = NewThreadGdbmDb()
- storeFile := path.Join(ms.dbDir, "state.gdbm")
- err := ms.stateStore.Connect(ctx, storeFile)
- if err != nil {
- return nil, err
- }
- return ms.stateStore, nil
-}
-
-func (ms *MenuStorageService) EnsureDbDir() error {
- err := os.MkdirAll(ms.dbDir, 0700)
- if err != nil {
- return fmt.Errorf("state dir create exited with error: %v\n", err)
- }
- return nil
-}
-
-func (ms *MenuStorageService) Close() error {
- errA := ms.stateStore.Close()
- errB := ms.userDataStore.Close()
- errC := ms.resourceStore.Close()
- if errA != nil || errB != nil || errC != nil {
- return fmt.Errorf("%v %v %v", errA, errB, errC)
- }
- return nil
-}
-
-
-
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
-}
-
-
package utils
-
-import (
- "encoding/binary"
-)
-
-type DataTyp uint16
-
-const (
- DATA_ACCOUNT DataTyp = iota
- DATA_ACCOUNT_CREATED
- DATA_TRACKING_ID
- DATA_PUBLIC_KEY
- DATA_CUSTODIAL_ID
- DATA_ACCOUNT_PIN
- DATA_ACCOUNT_STATUS
- DATA_FIRST_NAME
- DATA_FAMILY_NAME
- DATA_YOB
- DATA_LOCATION
- DATA_GENDER
- DATA_OFFERINGS
- DATA_RECIPIENT
- DATA_AMOUNT
- DATA_TEMPORARY_PIN
-)
-
-func typToBytes(typ DataTyp) []byte {
- var b [2]byte
- binary.BigEndian.PutUint16(b[:], uint16(typ))
- return b[:]
-}
-
-func PackKey(typ DataTyp, data []byte) []byte {
- v := typToBytes(typ)
- return append(v, data...)
-}
-
-
-
package utils
-
-var isoCodes = map[string]bool{
- "eng": true, // English
- "swa": true, // Swahili
-
-}
-
-func IsValidISO639(code string) bool {
- return isoCodes[code]
-}
-
-
-
package utils
-
-import (
- "context"
-
- "git.defalsify.org/vise.git/db"
-)
-
-type DataStore interface {
- db.Db
- ReadEntry(ctx context.Context, sessionId string, typ DataTyp) ([]byte, error)
- WriteEntry(ctx context.Context, sessionId string, typ DataTyp, value []byte) error
-}
-
-type UserDataStore struct {
- db.Db
-}
-
-// ReadEntry retrieves an entry from the store based on the provided parameters.
-func (store *UserDataStore) ReadEntry(ctx context.Context, sessionId string, typ DataTyp) ([]byte, error) {
- store.SetPrefix(db.DATATYPE_USERDATA)
- store.SetSession(sessionId)
- k := PackKey(typ, []byte(sessionId))
- return store.Get(ctx, k)
-}
-
-func (store *UserDataStore) WriteEntry(ctx context.Context, sessionId string, typ DataTyp, value []byte) error {
- store.SetPrefix(db.DATATYPE_USERDATA)
- store.SetSession(sessionId)
- k := PackKey(typ, []byte(sessionId))
- return store.Put(ctx, k, value)
-}
-
-
-
-
-
-