forked from urdt/ussd
2962 lines
113 KiB
HTML
2962 lines
113 KiB
HTML
|
|
||
|
<!DOCTYPE html>
|
||
|
<html>
|
||
|
<head>
|
||
|
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
|
||
|
<title>africastalking: Go Coverage Report</title>
|
||
|
<style>
|
||
|
body {
|
||
|
background: black;
|
||
|
color: rgb(80, 80, 80);
|
||
|
}
|
||
|
body, pre, #legend span {
|
||
|
font-family: Menlo, monospace;
|
||
|
font-weight: bold;
|
||
|
}
|
||
|
#topbar {
|
||
|
background: black;
|
||
|
position: fixed;
|
||
|
top: 0; left: 0; right: 0;
|
||
|
height: 42px;
|
||
|
border-bottom: 1px solid rgb(80, 80, 80);
|
||
|
}
|
||
|
#content {
|
||
|
margin-top: 50px;
|
||
|
}
|
||
|
#nav, #legend {
|
||
|
float: left;
|
||
|
margin-left: 10px;
|
||
|
}
|
||
|
#legend {
|
||
|
margin-top: 12px;
|
||
|
}
|
||
|
#nav {
|
||
|
margin-top: 10px;
|
||
|
}
|
||
|
#legend span {
|
||
|
margin: 0 5px;
|
||
|
}
|
||
|
.cov0 { color: rgb(192, 0, 0) }
|
||
|
.cov1 { color: rgb(128, 128, 128) }
|
||
|
.cov2 { color: rgb(116, 140, 131) }
|
||
|
.cov3 { color: rgb(104, 152, 134) }
|
||
|
.cov4 { color: rgb(92, 164, 137) }
|
||
|
.cov5 { color: rgb(80, 176, 140) }
|
||
|
.cov6 { color: rgb(68, 188, 143) }
|
||
|
.cov7 { color: rgb(56, 200, 146) }
|
||
|
.cov8 { color: rgb(44, 212, 149) }
|
||
|
.cov9 { color: rgb(32, 224, 152) }
|
||
|
.cov10 { color: rgb(20, 236, 155) }
|
||
|
|
||
|
</style>
|
||
|
</head>
|
||
|
<body>
|
||
|
<div id="topbar">
|
||
|
<div id="nav">
|
||
|
<select id="files">
|
||
|
|
||
|
<option value="file0">git.grassecon.net/urdt/ussd/cmd/africastalking/main.go (0.0%)</option>
|
||
|
|
||
|
<option value="file1">git.grassecon.net/urdt/ussd/cmd/async/main.go (0.0%)</option>
|
||
|
|
||
|
<option value="file2">git.grassecon.net/urdt/ussd/cmd/http/main.go (0.0%)</option>
|
||
|
|
||
|
<option value="file3">git.grassecon.net/urdt/ussd/cmd/main.go (0.0%)</option>
|
||
|
|
||
|
<option value="file4">git.grassecon.net/urdt/ussd/internal/handlers/base.go (0.0%)</option>
|
||
|
|
||
|
<option value="file5">git.grassecon.net/urdt/ussd/internal/handlers/handlerservice.go (0.0%)</option>
|
||
|
|
||
|
<option value="file6">git.grassecon.net/urdt/ussd/internal/handlers/server/accountservice.go (0.0%)</option>
|
||
|
|
||
|
<option value="file7">git.grassecon.net/urdt/ussd/internal/handlers/ussd/menuhandler.go (78.2%)</option>
|
||
|
|
||
|
<option value="file8">git.grassecon.net/urdt/ussd/internal/http/at_session_handler.go (86.7%)</option>
|
||
|
|
||
|
<option value="file9">git.grassecon.net/urdt/ussd/internal/http/server.go (88.1%)</option>
|
||
|
|
||
|
<option value="file10">git.grassecon.net/urdt/ussd/internal/mocks/dbmock.go (0.0%)</option>
|
||
|
|
||
|
<option value="file11">git.grassecon.net/urdt/ussd/internal/mocks/httpmocks/enginemock.go (25.0%)</option>
|
||
|
|
||
|
<option value="file12">git.grassecon.net/urdt/ussd/internal/mocks/httpmocks/requesthandlermock.go (71.4%)</option>
|
||
|
|
||
|
<option value="file13">git.grassecon.net/urdt/ussd/internal/mocks/httpmocks/requestparsermock.go (100.0%)</option>
|
||
|
|
||
|
<option value="file14">git.grassecon.net/urdt/ussd/internal/mocks/httpmocks/writermock.go (100.0%)</option>
|
||
|
|
||
|
<option value="file15">git.grassecon.net/urdt/ussd/internal/mocks/servicemock.go (100.0%)</option>
|
||
|
|
||
|
<option value="file16">git.grassecon.net/urdt/ussd/internal/mocks/userdbmock.go (100.0%)</option>
|
||
|
|
||
|
<option value="file17">git.grassecon.net/urdt/ussd/internal/storage/gdbm.go (0.0%)</option>
|
||
|
|
||
|
<option value="file18">git.grassecon.net/urdt/ussd/internal/storage/storage.go (0.0%)</option>
|
||
|
|
||
|
<option value="file19">git.grassecon.net/urdt/ussd/internal/storage/storageservice.go (0.0%)</option>
|
||
|
|
||
|
<option value="file20">git.grassecon.net/urdt/ussd/internal/utils/age.go (14.3%)</option>
|
||
|
|
||
|
<option value="file21">git.grassecon.net/urdt/ussd/internal/utils/db.go (0.0%)</option>
|
||
|
|
||
|
<option value="file22">git.grassecon.net/urdt/ussd/internal/utils/isocode.go (100.0%)</option>
|
||
|
|
||
|
<option value="file23">git.grassecon.net/urdt/ussd/internal/utils/userStore.go (0.0%)</option>
|
||
|
|
||
|
</select>
|
||
|
</div>
|
||
|
<div id="legend">
|
||
|
<span>not tracked</span>
|
||
|
|
||
|
<span class="cov0">not covered</span>
|
||
|
<span class="cov8">covered</span>
|
||
|
|
||
|
</div>
|
||
|
</div>
|
||
|
<div id="content">
|
||
|
|
||
|
<pre class="file" id="file0" style="display: none">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) <span class="cov0" title="0">{
|
||
|
rqv, ok := rq.(*http.Request)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return "", handlers.ErrInvalidRequest
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">if err := rqv.ParseForm(); err != nil </span><span class="cov0" title="0">{
|
||
|
return "", fmt.Errorf("failed to parse form data: %v", err)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">phoneNumber := rqv.FormValue("phoneNumber")
|
||
|
if phoneNumber == "" </span><span class="cov0" title="0">{
|
||
|
return "", fmt.Errorf("no phone number found")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">return phoneNumber, nil</span>
|
||
|
}
|
||
|
|
||
|
func (arp *atRequestParser) GetInput(rq any) ([]byte, error) <span class="cov0" title="0">{
|
||
|
rqv, ok := rq.(*http.Request)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return nil, handlers.ErrInvalidRequest
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">if err := rqv.ParseForm(); err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, fmt.Errorf("failed to parse form data: %v", err)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">text := rqv.FormValue("text")
|
||
|
|
||
|
parts := strings.Split(text, "*")
|
||
|
if len(parts) == 0 </span><span class="cov0" title="0">{
|
||
|
return nil, fmt.Errorf("no input found")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">return []byte(parts[len(parts)-1]), nil</span>
|
||
|
}
|
||
|
|
||
|
func main() <span class="cov0" title="0">{
|
||
|
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 </span><span class="cov0" title="0">{
|
||
|
cfg.EngineDebug = true
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">menuStorageService := storage.NewMenuStorageService(dbDir, resourceDir)
|
||
|
rs, err := menuStorageService.GetResource(ctx)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">err = menuStorageService.EnsureDbDir()
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">userdataStore, err := menuStorageService.GetUserdataDb(ctx)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">defer userdataStore.Close()
|
||
|
|
||
|
dbResource, ok := rs.(*resource.DbResource)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">lhs, err := handlers.NewLocalHandlerService(pfp, true, dbResource, cfg, rs)
|
||
|
lhs.SetDataStore(&userdataStore)
|
||
|
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">hl, err := lhs.GetHandler()
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">stateStore, err := menuStorageService.GetStateStore(ctx)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">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() </span><span class="cov0" title="0">{
|
||
|
select </span>{
|
||
|
case _ = <-cint:<span class="cov0" title="0"></span>
|
||
|
case _ = <-cterm:<span class="cov0" title="0"></span>
|
||
|
}
|
||
|
<span class="cov0" title="0">s.Shutdown(ctx)</span>
|
||
|
}()
|
||
|
<span class="cov0" title="0">err = s.ListenAndServe()
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
logg.Infof("Server closed with error", "err", err)
|
||
|
}</span>
|
||
|
}
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file1" style="display: none">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) <span class="cov0" title="0">{
|
||
|
return p.sessionId, nil
|
||
|
}</span>
|
||
|
|
||
|
func (p *asyncRequestParser) GetInput(r any) ([]byte, error) <span class="cov0" title="0">{
|
||
|
return p.input, nil
|
||
|
}</span>
|
||
|
|
||
|
func main() <span class="cov0" title="0">{
|
||
|
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 </span><span class="cov0" title="0">{
|
||
|
cfg.EngineDebug = true
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">menuStorageService := storage.NewMenuStorageService(dbDir, resourceDir)
|
||
|
rs, err := menuStorageService.GetResource(ctx)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">err = menuStorageService.EnsureDbDir()
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">userdataStore, err := menuStorageService.GetUserdataDb(ctx)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">defer userdataStore.Close()
|
||
|
|
||
|
dbResource, ok := rs.(*resource.DbResource)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">lhs, err := handlers.NewLocalHandlerService(pfp, true, dbResource, cfg, rs)
|
||
|
lhs.SetDataStore(&userdataStore)
|
||
|
|
||
|
hl, err := lhs.GetHandler()
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">stateStore, err := menuStorageService.GetStateStore(ctx)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">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() </span><span class="cov0" title="0">{
|
||
|
select </span>{
|
||
|
case _ = <-cint:<span class="cov0" title="0"></span>
|
||
|
case _ = <-cterm:<span class="cov0" title="0"></span>
|
||
|
}
|
||
|
<span class="cov0" title="0">sh.Shutdown()</span>
|
||
|
}()
|
||
|
|
||
|
<span class="cov0" title="0">for true </span><span class="cov0" title="0">{
|
||
|
rqs, err = sh.Process(rqs)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
logg.ErrorCtxf(ctx, "error in process: %v", "err", err)
|
||
|
fmt.Errorf("error in process: %v", err)
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">rqs, err = sh.Output(rqs)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
logg.ErrorCtxf(ctx, "error in output: %v", "err", err)
|
||
|
fmt.Errorf("error in output: %v", err)
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">rqs, err = sh.Reset(rqs)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
logg.ErrorCtxf(ctx, "error in reset: %v", "err", err)
|
||
|
fmt.Errorf("error in reset: %v", err)
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">fmt.Println("")
|
||
|
_, err = fmt.Scanln(&rqs.Input)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
logg.ErrorCtxf(ctx, "error in input", "err", err)
|
||
|
fmt.Errorf("error in input: %v", err)
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
}
|
||
|
}
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file2" style="display: none">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() <span class="cov0" title="0">{
|
||
|
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 </span><span class="cov0" title="0">{
|
||
|
cfg.EngineDebug = true
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">menuStorageService := storage.NewMenuStorageService(dbDir, resourceDir)
|
||
|
rs, err := menuStorageService.GetResource(ctx)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">err = menuStorageService.EnsureDbDir()
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">userdataStore, err := menuStorageService.GetUserdataDb(ctx)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">defer userdataStore.Close()
|
||
|
|
||
|
dbResource, ok := rs.(*resource.DbResource)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">lhs, err := handlers.NewLocalHandlerService(pfp, true, dbResource, cfg, rs)
|
||
|
lhs.SetDataStore(&userdataStore)
|
||
|
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">hl, err := lhs.GetHandler()
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">stateStore, err := menuStorageService.GetStateStore(ctx)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">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() </span><span class="cov0" title="0">{
|
||
|
select </span>{
|
||
|
case _ = <-cint:<span class="cov0" title="0"></span>
|
||
|
case _ = <-cterm:<span class="cov0" title="0"></span>
|
||
|
}
|
||
|
<span class="cov0" title="0">s.Shutdown(ctx)</span>
|
||
|
}()
|
||
|
<span class="cov0" title="0">err = s.ListenAndServe()
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
logg.Infof("Server closed with error", "err", err)
|
||
|
}</span>
|
||
|
}
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file3" style="display: none">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() <span class="cov0" title="0">{
|
||
|
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 </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">rs, err := menuStorageService.GetResource(ctx)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">pe, err := menuStorageService.GetPersister(ctx)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">userdatastore, err := menuStorageService.GetUserdataDb(ctx)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">dbResource, ok := rs.(*resource.DbResource)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">lhs, err := handlers.NewLocalHandlerService(pfp, true, dbResource, cfg, rs)
|
||
|
lhs.SetDataStore(&userdatastore)
|
||
|
lhs.SetPersister(pe)
|
||
|
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">hl, err := lhs.GetHandler()
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, err.Error())
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">en := lhs.GetEngine()
|
||
|
en = en.WithFirst(hl.Init)
|
||
|
if engineDebug </span><span class="cov0" title="0">{
|
||
|
en = en.WithDebug(nil)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">err = engine.Loop(ctx, en, os.Stdin, os.Stdout, nil)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Fprintf(os.Stderr, "loop exited with error: %v\n", err)
|
||
|
os.Exit(1)
|
||
|
}</span>
|
||
|
}
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file4" style="display: none">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 <span class="cov0" title="0">{
|
||
|
return &BaseSessionHandler{
|
||
|
cfgTemplate: cfg,
|
||
|
rs: rs,
|
||
|
hn: hn,
|
||
|
rp: rp,
|
||
|
provider: storage.NewSimpleStorageProvider(stateDb, userdataDb),
|
||
|
}
|
||
|
}</span>
|
||
|
|
||
|
func(f* BaseSessionHandler) Shutdown() <span class="cov0" title="0">{
|
||
|
err := f.provider.Close()
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
logg.Errorf("handler shutdown error", "err", err)
|
||
|
}</span>
|
||
|
}
|
||
|
|
||
|
func(f *BaseSessionHandler) GetEngine(cfg engine.Config, rs resource.Resource, pr *persist.Persister) engine.Engine <span class="cov0" title="0">{
|
||
|
en := engine.NewEngine(cfg, rs)
|
||
|
en = en.WithPersister(pr)
|
||
|
return en
|
||
|
}</span>
|
||
|
|
||
|
func(f *BaseSessionHandler) Process(rqs RequestSession) (RequestSession, error) <span class="cov0" title="0">{
|
||
|
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 </span><span class="cov0" title="0">{
|
||
|
logg.ErrorCtxf(rqs.Ctx, "", "storage get error", err)
|
||
|
return rqs, ErrStorage
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">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 </span><span class="cov0" title="0">{
|
||
|
perr := f.provider.Put(rqs.Config.SessionId, rqs.Storage)
|
||
|
rqs.Storage = nil
|
||
|
if perr != nil </span><span class="cov0" title="0">{
|
||
|
logg.ErrorCtxf(rqs.Ctx, "", "storage put error", perr)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">return rqs, ErrEngineType</span>
|
||
|
}
|
||
|
<span class="cov0" title="0">en = en.WithFirst(f.hn.Init)
|
||
|
if rqs.Config.EngineDebug </span><span class="cov0" title="0">{
|
||
|
en = en.WithDebug(nil)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">rqs.Engine = en
|
||
|
|
||
|
r, err = rqs.Engine.Exec(rqs.Ctx, rqs.Input)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
perr := f.provider.Put(rqs.Config.SessionId, rqs.Storage)
|
||
|
rqs.Storage = nil
|
||
|
if perr != nil </span><span class="cov0" title="0">{
|
||
|
logg.ErrorCtxf(rqs.Ctx, "", "storage put error", perr)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">return rqs, err</span>
|
||
|
}
|
||
|
|
||
|
<span class="cov0" title="0">rqs.Continue = r
|
||
|
return rqs, nil</span>
|
||
|
}
|
||
|
|
||
|
func(f *BaseSessionHandler) Output(rqs RequestSession) (RequestSession, error) <span class="cov0" title="0">{
|
||
|
var err error
|
||
|
_, err = rqs.Engine.Flush(rqs.Ctx, rqs.Writer)
|
||
|
return rqs, err
|
||
|
}</span>
|
||
|
|
||
|
func(f *BaseSessionHandler) Reset(rqs RequestSession) (RequestSession, error) <span class="cov0" title="0">{
|
||
|
defer f.provider.Put(rqs.Config.SessionId, rqs.Storage)
|
||
|
return rqs, rqs.Engine.Finish()
|
||
|
}</span>
|
||
|
|
||
|
func(f *BaseSessionHandler) GetConfig() engine.Config <span class="cov0" title="0">{
|
||
|
return f.cfgTemplate
|
||
|
}</span>
|
||
|
|
||
|
func(f *BaseSessionHandler) GetRequestParser() RequestParser <span class="cov0" title="0">{
|
||
|
return f.rp
|
||
|
}</span>
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file5" style="display: none">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) <span class="cov0" title="0">{
|
||
|
flagParser := asm.NewFlagParser().WithDebug()
|
||
|
_, err := flagParser.Load(fp)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">return flagParser, nil</span>
|
||
|
}
|
||
|
|
||
|
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) <span class="cov0" title="0">{
|
||
|
parser, err := getParser(fp, debug)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">return &LocalHandlerService{
|
||
|
Parser: parser,
|
||
|
DbRs: dbResource,
|
||
|
Cfg: cfg,
|
||
|
Rs: rs,
|
||
|
}, nil</span>
|
||
|
}
|
||
|
|
||
|
func (ls *LocalHandlerService) SetPersister(Pe *persist.Persister) <span class="cov0" title="0">{
|
||
|
ls.Pe = Pe
|
||
|
}</span>
|
||
|
|
||
|
func (ls *LocalHandlerService) SetDataStore(db *db.Db) <span class="cov0" title="0">{
|
||
|
ls.UserdataStore = db
|
||
|
}</span>
|
||
|
|
||
|
func (ls *LocalHandlerService) GetHandler() (*ussd.Handlers, error) <span class="cov0" title="0">{
|
||
|
ussdHandlers, err := ussd.NewHandlers(ls.Parser, *ls.UserdataStore)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">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</span>
|
||
|
}
|
||
|
|
||
|
// TODO: enable setting of sessionId on engine init time
|
||
|
func (ls *LocalHandlerService) GetEngine() *engine.DefaultEngine <span class="cov0" title="0">{
|
||
|
en := engine.NewEngine(ls.Cfg, ls.Rs)
|
||
|
en = en.WithPersister(ls.Pe)
|
||
|
return en
|
||
|
}</span>
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file6" style="display: none">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) <span class="cov0" title="0">{
|
||
|
resp, err := http.Get(config.TrackStatusURL + trackingId)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">defer resp.Body.Close()
|
||
|
|
||
|
body, err := io.ReadAll(resp.Body)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">var trackResp models.TrackStatusResponse
|
||
|
err = json.Unmarshal(body, &trackResp)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
|
||
|
// status := trackResp.Result.Transaction.Status
|
||
|
|
||
|
<span class="cov0" title="0">return &trackResp, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov0" title="0">{
|
||
|
|
||
|
resp, err := http.Get(config.BalanceURL + publicKey)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">defer resp.Body.Close()
|
||
|
|
||
|
body, err := io.ReadAll(resp.Body)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">var balanceResp models.BalanceResponse
|
||
|
err = json.Unmarshal(body, &balanceResp)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
|
||
|
//balance := balanceResp.Result.Balance
|
||
|
<span class="cov0" title="0">return &balanceResp, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov0" title="0">{
|
||
|
resp, err := http.Post(config.CreateAccountURL, "application/json", nil)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">defer resp.Body.Close()
|
||
|
|
||
|
body, err := io.ReadAll(resp.Body)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">var accountResp models.AccountResponse
|
||
|
err = json.Unmarshal(body, &accountResp)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">return &accountResp, nil</span>
|
||
|
}
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file7" style="display: none">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) <span class="cov8" title="1">{
|
||
|
parser := asm.NewFlagParser()
|
||
|
_, err := parser.Load(csvPath)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, fmt.Errorf("failed to load flag parser: %v", err)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">return &FlagManager{
|
||
|
parser: parser,
|
||
|
}, nil</span>
|
||
|
}
|
||
|
|
||
|
// GetFlag retrieves a flag value by its label
|
||
|
func (fm *FlagManager) GetFlag(label string) (uint32, error) <span class="cov8" title="1">{
|
||
|
return fm.parser.GetFlag(label)
|
||
|
}</span>
|
||
|
|
||
|
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) <span class="cov0" title="0">{
|
||
|
if userdataStore == nil </span><span class="cov0" title="0">{
|
||
|
return nil, fmt.Errorf("cannot create handler with nil userdata store")
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">userDb := &utils.UserDataStore{
|
||
|
Db: userdataStore,
|
||
|
}
|
||
|
h := &Handlers{
|
||
|
userdataStore: userDb,
|
||
|
flagManager: appFlags,
|
||
|
accountService: &server.AccountService{},
|
||
|
}
|
||
|
return h, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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 <span class="cov8" title="1">{
|
||
|
match, _ := regexp.MatchString(pinPattern, pin)
|
||
|
return match
|
||
|
}</span>
|
||
|
|
||
|
func (h *Handlers) WithPersister(pe *persist.Persister) *Handlers <span class="cov8" title="1">{
|
||
|
if h.pe != nil </span><span class="cov8" title="1">{
|
||
|
panic("persister already set")</span>
|
||
|
}
|
||
|
<span class="cov8" title="1">h.pe = pe
|
||
|
return h</span>
|
||
|
}
|
||
|
|
||
|
func (h *Handlers) Init(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov0" title="0">{
|
||
|
var r resource.Result
|
||
|
|
||
|
if h.pe == nil </span><span class="cov0" title="0">{
|
||
|
logg.WarnCtxf(ctx, "handler init called before it is ready or more than once", "state", h.st, "cache", h.ca)
|
||
|
return r, nil
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">h.st = h.pe.GetState()
|
||
|
h.ca = h.pe.GetMemory()
|
||
|
if h.st == nil || h.ca == nil </span><span class="cov0" title="0">{
|
||
|
logg.ErrorCtxf(ctx, "perister fail in handler", "state", h.st, "cache", h.ca)
|
||
|
return r, fmt.Errorf("cannot get state and memory for handler")
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">h.pe = nil
|
||
|
|
||
|
logg.DebugCtxf(ctx, "handler has been initialized", "state", h.st, "cache", h.ca)
|
||
|
|
||
|
return r, nil</span>
|
||
|
}
|
||
|
|
||
|
// SetLanguage sets the language across the menu
|
||
|
func (h *Handlers) SetLanguage(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
|
||
|
symbol, _ := h.st.Where()
|
||
|
code := strings.Split(symbol, "_")[1]
|
||
|
|
||
|
if !utils.IsValidISO639(code) </span><span class="cov0" title="0">{
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">res.FlagSet = append(res.FlagSet, state.FLAG_LANG)
|
||
|
res.Content = code
|
||
|
|
||
|
languageSetFlag, err := h.flagManager.GetFlag("flag_language_set")
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">res.FlagSet = append(res.FlagSet, languageSetFlag)
|
||
|
|
||
|
return res, nil</span>
|
||
|
}
|
||
|
|
||
|
func (h *Handlers) createAccountNoExist(ctx context.Context, sessionId string, res *resource.Result) error <span class="cov8" title="1">{
|
||
|
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 </span><span class="cov8" title="1">{
|
||
|
store := h.userdataStore
|
||
|
err := store.WriteEntry(ctx, sessionId, key, []byte(value))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return err
|
||
|
}</span>
|
||
|
}
|
||
|
<span class="cov8" title="1">flag_account_created, _ := h.flagManager.GetFlag("flag_account_created")
|
||
|
res.FlagSet = append(res.FlagSet, flag_account_created)
|
||
|
return err</span>
|
||
|
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">store := h.userdataStore
|
||
|
_, err = store.ReadEntry(ctx, sessionId, utils.DATA_ACCOUNT_CREATED)
|
||
|
if err != nil </span><span class="cov8" title="1">{
|
||
|
if db.IsNotFound(err) </span><span class="cov8" title="1">{
|
||
|
logg.Printf(logging.LVL_INFO, "Creating an account because it doesn't exist")
|
||
|
err = h.createAccountNoExist(ctx, sessionId, &res)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
}
|
||
|
}
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// SavePin persists the user's PIN choice into the filesystem
|
||
|
func (h *Handlers) SavePin(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">flag_incorrect_pin, _ := h.flagManager.GetFlag("flag_incorrect_pin")
|
||
|
|
||
|
accountPIN := string(input)
|
||
|
// Validate that the PIN is a 4-digit number
|
||
|
if !isValidPIN(accountPIN) </span><span class="cov8" title="1">{
|
||
|
res.FlagSet = append(res.FlagSet, flag_incorrect_pin)
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">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 </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
func (h *Handlers) VerifyNewPin(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
res := resource.Result{}
|
||
|
_, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">flag_valid_pin, _ := h.flagManager.GetFlag("flag_valid_pin")
|
||
|
pinInput := string(input)
|
||
|
// Validate that the PIN is a 4-digit number
|
||
|
if isValidPIN(pinInput) </span><span class="cov8" title="1">{
|
||
|
res.FlagSet = append(res.FlagSet, flag_valid_pin)
|
||
|
}</span> else<span class="cov8" title="1"> {
|
||
|
res.FlagReset = append(res.FlagReset, flag_valid_pin)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
func (h *Handlers) SaveTemporaryPin(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">flag_incorrect_pin, _ := h.flagManager.GetFlag("flag_incorrect_pin")
|
||
|
|
||
|
accountPIN := string(input)
|
||
|
|
||
|
// Validate that the PIN is a 4-digit number
|
||
|
if !isValidPIN(accountPIN) </span><span class="cov0" title="0">{
|
||
|
res.FlagSet = append(res.FlagSet, flag_incorrect_pin)
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">store := h.userdataStore
|
||
|
err = store.WriteEntry(ctx, sessionId, utils.DATA_TEMPORARY_PIN, []byte(accountPIN))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
func (h *Handlers) ConfirmPinChange(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">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 </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">if bytes.Equal(temporaryPin, input) </span><span class="cov8" title="1">{
|
||
|
res.FlagReset = append(res.FlagReset, flag_pin_mismatch)
|
||
|
}</span> else<span class="cov0" title="0"> {
|
||
|
res.FlagSet = append(res.FlagSet, flag_pin_mismatch)
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">err = store.WriteEntry(ctx, sessionId, utils.DATA_ACCOUNT_PIN, []byte(temporaryPin))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
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 </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">store := h.userdataStore
|
||
|
AccountPin, err := store.ReadEntry(ctx, sessionId, utils.DATA_ACCOUNT_PIN)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">if bytes.Equal(input, AccountPin) </span><span class="cov8" title="1">{
|
||
|
res.FlagSet = []uint32{flag_valid_pin}
|
||
|
res.FlagReset = []uint32{flag_pin_mismatch}
|
||
|
res.FlagSet = append(res.FlagSet, flag_pin_set)
|
||
|
}</span> else<span class="cov8" title="1"> {
|
||
|
res.FlagSet = []uint32{flag_pin_mismatch}
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// codeFromCtx retrieves language codes from the context that can be used for handling translations
|
||
|
func codeFromCtx(ctx context.Context) string <span class="cov8" title="1">{
|
||
|
var code string
|
||
|
if ctx.Value("Language") != nil </span><span class="cov0" title="0">{
|
||
|
lang := ctx.Value("Language").(lang.Language)
|
||
|
code = lang.Code
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">return code</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">if len(input) > 0 </span><span class="cov8" title="1">{
|
||
|
firstName := string(input)
|
||
|
store := h.userdataStore
|
||
|
err = store.WriteEntry(ctx, sessionId, utils.DATA_FIRST_NAME, []byte(firstName))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
}
|
||
|
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">if len(input) > 0 </span><span class="cov8" title="1">{
|
||
|
familyName := string(input)
|
||
|
store := h.userdataStore
|
||
|
err = store.WriteEntry(ctx, sessionId, utils.DATA_FAMILY_NAME, []byte(familyName))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
} else<span class="cov0" title="0"> {
|
||
|
return res, fmt.Errorf("a family name cannot be less than one character")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">if len(input) == 4 </span><span class="cov8" title="1">{
|
||
|
yob := string(input)
|
||
|
store := h.userdataStore
|
||
|
err = store.WriteEntry(ctx, sessionId, utils.DATA_YOB, []byte(yob))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
}
|
||
|
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">if len(input) > 0 </span><span class="cov8" title="1">{
|
||
|
location := string(input)
|
||
|
store := h.userdataStore
|
||
|
err = store.WriteEntry(ctx, sessionId, utils.DATA_LOCATION, []byte(location))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
}
|
||
|
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
symbol, _ := h.st.Where()
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">gender := strings.Split(symbol, "_")[1]
|
||
|
store := h.userdataStore
|
||
|
err = store.WriteEntry(ctx, sessionId, utils.DATA_GENDER, []byte(gender))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">if len(input) > 0 </span><span class="cov8" title="1">{
|
||
|
offerings := string(input)
|
||
|
store := h.userdataStore
|
||
|
err = store.WriteEntry(ctx, sessionId, utils.DATA_OFFERINGS, []byte(offerings))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
}
|
||
|
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
|
||
|
flag_allow_update, _ := h.flagManager.GetFlag("flag_allow_update")
|
||
|
|
||
|
res.FlagReset = append(res.FlagReset, flag_allow_update)
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
|
||
|
flag_account_authorized, _ := h.flagManager.GetFlag("flag_account_authorized")
|
||
|
|
||
|
res.FlagReset = append(res.FlagReset, flag_account_authorized)
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
|
||
|
// CheckIdentifier retrieves the PublicKey from the JSON data file.
|
||
|
func (h *Handlers) CheckIdentifier(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">store := h.userdataStore
|
||
|
publicKey, _ := store.ReadEntry(ctx, sessionId, utils.DATA_PUBLIC_KEY)
|
||
|
|
||
|
res.Content = string(publicKey)
|
||
|
|
||
|
return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">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 </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">if len(input) == 4 </span><span class="cov8" title="1">{
|
||
|
if bytes.Equal(input, AccountPin) </span><span class="cov8" title="1">{
|
||
|
if h.st.MatchFlag(flag_account_authorized, false) </span><span class="cov8" title="1">{
|
||
|
res.FlagReset = append(res.FlagReset, flag_incorrect_pin)
|
||
|
res.FlagSet = append(res.FlagSet, flag_allow_update, flag_account_authorized)
|
||
|
}</span> else<span class="cov0" title="0"> {
|
||
|
res.FlagSet = append(res.FlagSet, flag_allow_update)
|
||
|
res.FlagReset = append(res.FlagReset, flag_account_authorized)
|
||
|
}</span>
|
||
|
} else<span class="cov8" title="1"> {
|
||
|
res.FlagSet = append(res.FlagSet, flag_incorrect_pin)
|
||
|
res.FlagReset = append(res.FlagReset, flag_account_authorized)
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
} else<span class="cov8" title="1"> {
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
flag_incorrect_pin, _ := h.flagManager.GetFlag("flag_incorrect_pin")
|
||
|
res.FlagReset = append(res.FlagReset, flag_incorrect_pin)
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
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 </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">store := h.userdataStore
|
||
|
trackingId, err := store.ReadEntry(ctx, sessionId, utils.DATA_TRACKING_ID)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">accountStatus, err := h.accountService.CheckAccountStatus(string(trackingId))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
fmt.Println("Error checking account status:", err)
|
||
|
return res, err
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">if !accountStatus.Ok </span><span class="cov8" title="1">{
|
||
|
res.FlagSet = append(res.FlagSet, flag_api_error)
|
||
|
return res, err
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">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 </span><span class="cov0" title="0">{
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">if accountStatus.Result.Transaction.Status == "SUCCESS" </span><span class="cov8" title="1">{
|
||
|
res.FlagSet = append(res.FlagSet, flag_account_success)
|
||
|
res.FlagReset = append(res.FlagReset, flag_account_pending)
|
||
|
}</span> else<span class="cov8" title="1"> {
|
||
|
res.FlagReset = append(res.FlagReset, flag_account_success)
|
||
|
res.FlagSet = append(res.FlagSet, flag_account_pending)
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// Quit displays the Thank you message and exits the menu
|
||
|
func (h *Handlers) Quit(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
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
|
||
|
}</span>
|
||
|
|
||
|
// QuitWithHelp displays helpline information then exits the menu
|
||
|
func (h *Handlers) QuitWithHelp(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov0" title="0">{
|
||
|
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
|
||
|
}</span>
|
||
|
|
||
|
// VerifyYob verifies the length of the given input
|
||
|
func (h *Handlers) VerifyYob(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
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 </span><span class="cov8" title="1">{
|
||
|
// If conversion fails, input is not numeric
|
||
|
res.FlagSet = append(res.FlagSet, flag_incorrect_date_format)
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">if len(date) == 4 </span><span class="cov8" title="1">{
|
||
|
res.FlagReset = append(res.FlagReset, flag_incorrect_date_format)
|
||
|
}</span> else<span class="cov8" title="1"> {
|
||
|
res.FlagSet = append(res.FlagSet, flag_incorrect_date_format)
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
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
|
||
|
}</span>
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
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 </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">store := h.userdataStore
|
||
|
publicKey, err := store.ReadEntry(ctx, sessionId, utils.DATA_PUBLIC_KEY)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">balanceResponse, err := h.accountService.CheckBalance(string(publicKey))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">if !balanceResponse.Ok </span><span class="cov8" title="1">{
|
||
|
res.FlagSet = append(res.FlagSet, flag_api_error)
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">res.FlagReset = append(res.FlagReset, flag_api_error)
|
||
|
balance := balanceResponse.Result.Balance
|
||
|
res.Content = balance
|
||
|
|
||
|
return res, nil</span>
|
||
|
}
|
||
|
|
||
|
func (h *Handlers) FetchCustodialBalances(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
flag_api_error, _ := h.flagManager.GetFlag("flag_api_call_error")
|
||
|
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">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 </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">balanceResponse, err := h.accountService.CheckBalance(string(publicKey))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">if !balanceResponse.Ok </span><span class="cov8" title="1">{
|
||
|
res.FlagSet = append(res.FlagSet, flag_api_error)
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">res.FlagReset = append(res.FlagReset, flag_api_error)
|
||
|
balance := balanceResponse.Result.Balance
|
||
|
|
||
|
switch balanceType </span>{
|
||
|
case "my":<span class="cov0" title="0">
|
||
|
res.Content = fmt.Sprintf("Your balance is %s", balance)</span>
|
||
|
case "community":<span class="cov0" title="0">
|
||
|
res.Content = fmt.Sprintf("Your community balance is %s", balance)</span>
|
||
|
default:<span class="cov8" title="1">
|
||
|
break</span>
|
||
|
}
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">recipient := string(input)
|
||
|
|
||
|
flag_invalid_recipient, _ := h.flagManager.GetFlag("flag_invalid_recipient")
|
||
|
|
||
|
if recipient != "0" </span><span class="cov8" title="1">{
|
||
|
// mimic invalid number check
|
||
|
if recipient == "000" </span><span class="cov8" title="1">{
|
||
|
res.FlagSet = append(res.FlagSet, flag_invalid_recipient)
|
||
|
res.Content = recipient
|
||
|
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">store := h.userdataStore
|
||
|
err = store.WriteEntry(ctx, sessionId, utils.DATA_RECIPIENT, []byte(recipient))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
}
|
||
|
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">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 </span><span class="cov0" title="0">{
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">err = store.WriteEntry(ctx, sessionId, utils.DATA_RECIPIENT, []byte(""))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">res.FlagReset = append(res.FlagReset, flag_invalid_recipient, flag_invalid_recipient_with_invite)
|
||
|
|
||
|
return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// ResetTransactionAmount resets the transaction amount and invalid flag
|
||
|
func (h *Handlers) ResetTransactionAmount(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">flag_invalid_amount, _ := h.flagManager.GetFlag("flag_invalid_amount")
|
||
|
store := h.userdataStore
|
||
|
err = store.WriteEntry(ctx, sessionId, utils.DATA_AMOUNT, []byte(""))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">res.FlagReset = append(res.FlagReset, flag_invalid_amount)
|
||
|
|
||
|
return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">store := h.userdataStore
|
||
|
publicKey, _ := store.ReadEntry(ctx, sessionId, utils.DATA_PUBLIC_KEY)
|
||
|
|
||
|
balanceResp, err := h.accountService.CheckBalance(string(publicKey))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">balance := balanceResp.Result.Balance
|
||
|
|
||
|
res.Content = balance
|
||
|
|
||
|
return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">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 </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">res.Content = balanceStr
|
||
|
|
||
|
// Parse the balance
|
||
|
balanceParts := strings.Split(balanceStr, " ")
|
||
|
if len(balanceParts) != 2 </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("unexpected balance format: %s", balanceStr)
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">balanceValue, err := strconv.ParseFloat(balanceParts[0], 64)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("failed to parse balance: %v", err)
|
||
|
}</span>
|
||
|
|
||
|
// Extract numeric part from input
|
||
|
<span class="cov8" title="1">re := regexp.MustCompile(`^(\d+(\.\d+)?)\s*(?:CELO)?$`)
|
||
|
matches := re.FindStringSubmatch(strings.TrimSpace(amountStr))
|
||
|
if len(matches) < 2 </span><span class="cov8" title="1">{
|
||
|
res.FlagSet = append(res.FlagSet, flag_invalid_amount)
|
||
|
res.Content = amountStr
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">inputAmount, err := strconv.ParseFloat(matches[1], 64)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
res.FlagSet = append(res.FlagSet, flag_invalid_amount)
|
||
|
res.Content = amountStr
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">if inputAmount > balanceValue </span><span class="cov8" title="1">{
|
||
|
res.FlagSet = append(res.FlagSet, flag_invalid_amount)
|
||
|
res.Content = amountStr
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">res.Content = fmt.Sprintf("%.3f", inputAmount) // Format to 3 decimal places
|
||
|
err = store.WriteEntry(ctx, sessionId, utils.DATA_AMOUNT, []byte(amountStr))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// GetRecipient returns the transaction recipient from the gdbm.
|
||
|
func (h *Handlers) GetRecipient(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">store := h.userdataStore
|
||
|
recipient, _ := store.ReadEntry(ctx, sessionId, utils.DATA_RECIPIENT)
|
||
|
|
||
|
res.Content = string(recipient)
|
||
|
|
||
|
return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// GetSender retrieves the public key from the Gdbm Db
|
||
|
func (h *Handlers) GetSender(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">store := h.userdataStore
|
||
|
publicKey, _ := store.ReadEntry(ctx, sessionId, utils.DATA_PUBLIC_KEY)
|
||
|
|
||
|
res.Content = string(publicKey)
|
||
|
|
||
|
return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// GetAmount retrieves the amount from teh Gdbm Db
|
||
|
func (h *Handlers) GetAmount(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">store := h.userdataStore
|
||
|
amount, _ := store.ReadEntry(ctx, sessionId, utils.DATA_AMOUNT)
|
||
|
|
||
|
res.Content = string(amount)
|
||
|
|
||
|
return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov0" title="0">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov0" title="0">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 </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">balance, err := h.accountService.CheckBalance(string(publicKey))
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return res, nil
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">res.Content = l.Get("Your account balance is %s", balance)
|
||
|
res.FlagReset = append(res.FlagReset, flag_account_authorized)
|
||
|
return res, nil</span>
|
||
|
}
|
||
|
|
||
|
// 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) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var err error
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">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 </span><span class="cov0" title="0">{
|
||
|
return res, err
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">res.FlagReset = append(res.FlagReset, account_authorized_flag)
|
||
|
return res, nil</span>
|
||
|
}
|
||
|
|
||
|
func (h *Handlers) GetProfileInfo(ctx context.Context, sym string, input []byte) (resource.Result, error) <span class="cov8" title="1">{
|
||
|
var res resource.Result
|
||
|
var defaultValue string
|
||
|
sessionId, ok := ctx.Value("SessionId").(string)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("missing session")
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">language, ok := ctx.Value("Language").(lang.Language)
|
||
|
if !ok </span><span class="cov0" title="0">{
|
||
|
return res, fmt.Errorf("value for 'Language' is not of type lang.Language")
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">code := language.Code
|
||
|
if code == "swa" </span><span class="cov8" title="1">{
|
||
|
defaultValue = "Haipo"
|
||
|
}</span> else<span class="cov8" title="1"> {
|
||
|
defaultValue = "Not Provided"
|
||
|
}</span>
|
||
|
|
||
|
// Helper function to handle nil byte slices and convert them to string
|
||
|
<span class="cov8" title="1">getEntryOrDefault := func(entry []byte, err error) string </span><span class="cov8" title="1">{
|
||
|
if err != nil || entry == nil </span><span class="cov0" title="0">{
|
||
|
return defaultValue
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">return string(entry)</span>
|
||
|
}
|
||
|
<span class="cov8" title="1">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 </span><span class="cov8" title="1">{
|
||
|
if firstName == defaultValue </span><span class="cov0" title="0">{
|
||
|
name = familyName
|
||
|
}</span> else<span class="cov8" title="1"> {
|
||
|
name = firstName + " " + familyName
|
||
|
}</span>
|
||
|
}
|
||
|
|
||
|
// Calculate age from year of birth
|
||
|
<span class="cov8" title="1">age := defaultValue
|
||
|
if yob != defaultValue </span><span class="cov8" title="1">{
|
||
|
if yobInt, err := strconv.Atoi(yob); err == nil </span><span class="cov8" title="1">{
|
||
|
age = strconv.Itoa(utils.CalculateAgeWithYOB(yobInt))
|
||
|
}</span> else<span class="cov0" title="0"> {
|
||
|
return res, fmt.Errorf("invalid year of birth: %v", err)
|
||
|
}</span>
|
||
|
}
|
||
|
<span class="cov8" title="1">switch language.Code </span>{
|
||
|
case "eng":<span class="cov8" title="1">
|
||
|
res.Content = fmt.Sprintf(
|
||
|
"Name: %s\nGender: %s\nAge: %s\nLocation: %s\nYou provide: %s\n",
|
||
|
name, gender, age, location, offerings,
|
||
|
)</span>
|
||
|
case "swa":<span class="cov8" title="1">
|
||
|
res.Content = fmt.Sprintf(
|
||
|
"Jina: %s\nJinsia: %s\nUmri: %s\nEneo: %s\nUnauza: %s\n",
|
||
|
name, gender, age, location, offerings,
|
||
|
)</span>
|
||
|
default:<span class="cov8" title="1">
|
||
|
res.Content = fmt.Sprintf(
|
||
|
"Name: %s\nGender: %s\nAge: %s\nLocation: %s\nYou provide: %s\n",
|
||
|
name, gender, age, location, offerings,
|
||
|
)</span>
|
||
|
}
|
||
|
|
||
|
<span class="cov8" title="1">return res, nil</span>
|
||
|
}
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file8" style="display: none">package http
|
||
|
|
||
|
import (
|
||
|
"io"
|
||
|
"net/http"
|
||
|
|
||
|
"git.grassecon.net/urdt/ussd/internal/handlers"
|
||
|
)
|
||
|
|
||
|
type ATSessionHandler struct {
|
||
|
*SessionHandler
|
||
|
}
|
||
|
|
||
|
func NewATSessionHandler(h handlers.RequestHandler) *ATSessionHandler <span class="cov8" title="1">{
|
||
|
return &ATSessionHandler{
|
||
|
SessionHandler: ToSessionHandler(h),
|
||
|
}
|
||
|
}</span>
|
||
|
|
||
|
func (ash *ATSessionHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) <span class="cov8" title="1">{
|
||
|
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 </span><span class="cov8" title="1">{
|
||
|
logg.ErrorCtxf(rqs.Ctx, "", "header processing error", err)
|
||
|
ash.writeError(w, 400, err)
|
||
|
return
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">rqs.Config = cfg
|
||
|
rqs.Input, err = rp.GetInput(req)
|
||
|
if err != nil </span><span class="cov8" title="1">{
|
||
|
logg.ErrorCtxf(rqs.Ctx, "", "header processing error", err)
|
||
|
ash.writeError(w, 400, err)
|
||
|
return
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">rqs, err = ash.Process(rqs)
|
||
|
switch err </span>{
|
||
|
case nil:<span class="cov8" title="1"> // set code to 200 if no err
|
||
|
code = 200</span>
|
||
|
case handlers.ErrStorage, handlers.ErrEngineInit, handlers.ErrEngineExec, handlers.ErrEngineType:<span class="cov8" title="1">
|
||
|
code = 500</span>
|
||
|
default:<span class="cov0" title="0">
|
||
|
code = 500</span>
|
||
|
}
|
||
|
|
||
|
<span class="cov8" title="1">if code != 200 </span><span class="cov8" title="1">{
|
||
|
ash.writeError(w, 500, err)
|
||
|
return
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">w.WriteHeader(200)
|
||
|
w.Header().Set("Content-Type", "text/plain")
|
||
|
rqs, err = ash.Output(rqs)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
ash.writeError(w, 500, err)
|
||
|
return
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">rqs, err = ash.Reset(rqs)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
ash.writeError(w, 500, err)
|
||
|
return
|
||
|
}</span>
|
||
|
}
|
||
|
|
||
|
func (ash *ATSessionHandler) Output(rqs handlers.RequestSession) (handlers.RequestSession, error) <span class="cov8" title="1">{
|
||
|
var err error
|
||
|
var prefix string
|
||
|
|
||
|
if rqs.Continue </span><span class="cov8" title="1">{
|
||
|
prefix = "CON "
|
||
|
}</span> else<span class="cov8" title="1"> {
|
||
|
prefix = "END "
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">_, err = io.WriteString(rqs.Writer, prefix)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return rqs, err
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">_, err = rqs.Engine.Flush(rqs.Ctx, rqs.Writer)
|
||
|
return rqs, err</span>
|
||
|
}</pre>
|
||
|
|
||
|
<pre class="file" id="file9" style="display: none">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) <span class="cov8" title="1">{
|
||
|
rqv, ok := rq.(*http.Request)
|
||
|
if !ok </span><span class="cov8" title="1">{
|
||
|
return "", handlers.ErrInvalidRequest
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">v := rqv.Header.Get("X-Vise-Session")
|
||
|
if v == "" </span><span class="cov8" title="1">{
|
||
|
return "", handlers.ErrSessionMissing
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">return v, nil</span>
|
||
|
}
|
||
|
|
||
|
func(rp *DefaultRequestParser) GetInput(rq any) ([]byte, error) <span class="cov8" title="1">{
|
||
|
rqv, ok := rq.(*http.Request)
|
||
|
if !ok </span><span class="cov8" title="1">{
|
||
|
return nil, handlers.ErrInvalidRequest
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">defer rqv.Body.Close()
|
||
|
v, err := ioutil.ReadAll(rqv.Body)
|
||
|
if err != nil </span><span class="cov8" title="1">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">return v, nil</span>
|
||
|
}
|
||
|
|
||
|
type SessionHandler struct {
|
||
|
handlers.RequestHandler
|
||
|
}
|
||
|
|
||
|
func ToSessionHandler(h handlers.RequestHandler) *SessionHandler <span class="cov8" title="1">{
|
||
|
return &SessionHandler{
|
||
|
RequestHandler: h,
|
||
|
}
|
||
|
}</span>
|
||
|
|
||
|
func(f *SessionHandler) writeError(w http.ResponseWriter, code int, err error) <span class="cov8" title="1">{
|
||
|
s := err.Error()
|
||
|
w.Header().Set("Content-Length", strconv.Itoa(len(s)))
|
||
|
w.WriteHeader(code)
|
||
|
_, err = w.Write([]byte{})
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
logg.Errorf("error writing error!!", "err", err, "olderr", s)
|
||
|
w.WriteHeader(500)
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">return</span>
|
||
|
}
|
||
|
|
||
|
func(f *SessionHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) <span class="cov8" title="1">{
|
||
|
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 </span><span class="cov8" title="1">{
|
||
|
logg.ErrorCtxf(rqs.Ctx, "", "header processing error", err)
|
||
|
f.writeError(w, 400, err)
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">rqs.Config = cfg
|
||
|
rqs.Input, err = rp.GetInput(req)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
logg.ErrorCtxf(rqs.Ctx, "", "header processing error", err)
|
||
|
f.writeError(w, 400, err)
|
||
|
return
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">rqs, err = f.Process(rqs)
|
||
|
switch err </span>{
|
||
|
case handlers.ErrStorage:<span class="cov8" title="1">
|
||
|
code = 500</span>
|
||
|
case handlers.ErrEngineInit:<span class="cov0" title="0">
|
||
|
code = 500</span>
|
||
|
case handlers.ErrEngineExec:<span class="cov0" title="0">
|
||
|
code = 500</span>
|
||
|
default:<span class="cov8" title="1">
|
||
|
code = 200</span>
|
||
|
}
|
||
|
|
||
|
<span class="cov8" title="1">if code != 200 </span><span class="cov8" title="1">{
|
||
|
f.writeError(w, 500, err)
|
||
|
return
|
||
|
}</span>
|
||
|
|
||
|
<span class="cov8" title="1">w.WriteHeader(200)
|
||
|
w.Header().Set("Content-Type", "text/plain")
|
||
|
rqs, err = f.Output(rqs)
|
||
|
rqs, perr = f.Reset(rqs)
|
||
|
if err != nil </span><span class="cov8" title="1">{
|
||
|
f.writeError(w, 500, err)
|
||
|
return
|
||
|
}</span>
|
||
|
<span class="cov8" title="1">if perr != nil </span><span class="cov8" title="1">{
|
||
|
f.writeError(w, 500, perr)
|
||
|
return
|
||
|
}</span>
|
||
|
}
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file10" style="display: none">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) <span class="cov0" title="0">{
|
||
|
m.Called(prefix)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockDb) Prefix() uint8 <span class="cov0" title="0">{
|
||
|
args := m.Called()
|
||
|
return args.Get(0).(uint8)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockDb) Safe() bool <span class="cov0" title="0">{
|
||
|
args := m.Called()
|
||
|
return args.Get(0).(bool)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockDb) SetLanguage(language *lang.Language) <span class="cov0" title="0">{
|
||
|
m.Called(language)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockDb) SetLock(uint8, bool) error <span class="cov0" title="0">{
|
||
|
args := m.Called()
|
||
|
return args.Error(0)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockDb) Connect(ctx context.Context, connectionStr string) error <span class="cov0" title="0">{
|
||
|
args := m.Called(ctx, connectionStr)
|
||
|
return args.Error(0)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockDb) SetSession(sessionId string) <span class="cov0" title="0">{
|
||
|
m.Called(sessionId)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockDb) Put(ctx context.Context, key, value []byte) error <span class="cov0" title="0">{
|
||
|
args := m.Called(ctx, key, value)
|
||
|
return args.Error(0)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockDb) Get(ctx context.Context, key []byte) ([]byte, error) <span class="cov0" title="0">{
|
||
|
args := m.Called(ctx, key)
|
||
|
return nil, args.Error(0)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockDb) Close() error <span class="cov0" title="0">{
|
||
|
args := m.Called(nil)
|
||
|
return args.Error(0)
|
||
|
}</span>
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file11" style="display: none">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) <span class="cov0" title="0">{
|
||
|
return m.InitFunc(ctx)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockEngine) Exec(ctx context.Context, input []byte) (bool, error) <span class="cov0" title="0">{
|
||
|
return m.ExecFunc(ctx, input)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockEngine) Flush(ctx context.Context, w io.Writer) (int, error) <span class="cov8" title="1">{
|
||
|
return m.FlushFunc(ctx, w)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockEngine) Finish() error <span class="cov0" title="0">{
|
||
|
return m.FinishFunc()
|
||
|
}</span>
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file12" style="display: none">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) <span class="cov8" title="1">{
|
||
|
return m.ProcessFunc(rqs)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockRequestHandler) GetConfig() engine.Config <span class="cov8" title="1">{
|
||
|
return m.GetConfigFunc()
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockRequestHandler) GetEngine(cfg engine.Config, rs resource.Resource, pe *persist.Persister) engine.Engine <span class="cov0" title="0">{
|
||
|
return m.GetEngineFunc(cfg, rs, pe)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockRequestHandler) Output(rs handlers.RequestSession) (handlers.RequestSession, error) <span class="cov8" title="1">{
|
||
|
return m.OutputFunc(rs)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockRequestHandler) Reset(rs handlers.RequestSession) (handlers.RequestSession, error) <span class="cov8" title="1">{
|
||
|
return m.ResetFunc(rs)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockRequestHandler) Shutdown() <span class="cov0" title="0">{
|
||
|
m.ShutdownFunc()
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockRequestHandler) GetRequestParser() handlers.RequestParser <span class="cov8" title="1">{
|
||
|
return m.GetRequestParserFunc()
|
||
|
}</span>
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file13" style="display: none">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) <span class="cov8" title="1">{
|
||
|
return m.GetSessionIdFunc(rq)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockRequestParser) GetInput(rq any) ([]byte, error) <span class="cov8" title="1">{
|
||
|
return m.GetInputFunc(rq)
|
||
|
}</span>
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file14" style="display: none">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) <span class="cov8" title="1">{
|
||
|
return len(p), nil
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockWriter) WriteString(s string) (n int, err error) <span class="cov8" title="1">{
|
||
|
m.WriteStringCalled = true
|
||
|
m.WrittenString = s
|
||
|
return len(s), nil
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockWriter) Header() http.Header <span class="cov8" title="1">{
|
||
|
return http.Header{}
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockWriter) WriteHeader(statusCode int) {<span class="cov8" title="1">}</pre>
|
||
|
|
||
|
<pre class="file" id="file15" style="display: none">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) <span class="cov8" title="1">{
|
||
|
args := m.Called()
|
||
|
return args.Get(0).(*models.AccountResponse), args.Error(1)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockAccountService) CheckBalance(publicKey string) (*models.BalanceResponse, error) <span class="cov8" title="1">{
|
||
|
args := m.Called(publicKey)
|
||
|
return args.Get(0).(*models.BalanceResponse), args.Error(1)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockAccountService) CheckAccountStatus(trackingId string) (*models.TrackStatusResponse, error) <span class="cov8" title="1">{
|
||
|
args := m.Called(trackingId)
|
||
|
return args.Get(0).(*models.TrackStatusResponse), args.Error(1)
|
||
|
}</pre>
|
||
|
|
||
|
<pre class="file" id="file16" style="display: none">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) <span class="cov8" title="1">{
|
||
|
args := m.Called(ctx, sessionId, typ)
|
||
|
return args.Get(0).([]byte), args.Error(1)
|
||
|
}</span>
|
||
|
|
||
|
func (m *MockUserDataStore) WriteEntry(ctx context.Context, sessionId string, typ utils.DataTyp, value []byte) error <span class="cov8" title="1">{
|
||
|
args := m.Called(ctx, sessionId, typ, value)
|
||
|
return args.Error(0)
|
||
|
}</span>
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file17" style="display: none">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 <span class="cov0" title="0">{
|
||
|
if dbC == nil </span><span class="cov0" title="0">{
|
||
|
dbC = make(map[string]chan db.Db)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">return &ThreadGdbmDb{}</span>
|
||
|
}
|
||
|
|
||
|
func(tdb *ThreadGdbmDb) Connect(ctx context.Context, connStr string) error <span class="cov0" title="0">{
|
||
|
var ok bool
|
||
|
_, ok = dbC[connStr]
|
||
|
if ok </span><span class="cov0" title="0">{
|
||
|
logg.WarnCtxf(ctx, "already registered thread gdbm, skipping", "connStr", connStr)
|
||
|
return nil
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">gdb := gdbmdb.NewGdbmDb()
|
||
|
err := gdb.Connect(ctx, connStr)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return err
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">dbC[connStr] = make(chan db.Db, 1)
|
||
|
dbC[connStr]<- gdb
|
||
|
tdb.connStr = connStr
|
||
|
return nil</span>
|
||
|
}
|
||
|
|
||
|
func(tdb *ThreadGdbmDb) reserve() <span class="cov0" title="0">{
|
||
|
if tdb.db == nil </span><span class="cov0" title="0">{
|
||
|
tdb.db = <-dbC[tdb.connStr]
|
||
|
}</span>
|
||
|
}
|
||
|
|
||
|
func(tdb *ThreadGdbmDb) release() <span class="cov0" title="0">{
|
||
|
if tdb.db == nil </span><span class="cov0" title="0">{
|
||
|
return
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">dbC[tdb.connStr] <- tdb.db
|
||
|
tdb.db = nil</span>
|
||
|
}
|
||
|
|
||
|
func(tdb *ThreadGdbmDb) SetPrefix(pfx uint8) <span class="cov0" title="0">{
|
||
|
tdb.reserve()
|
||
|
tdb.db.SetPrefix(pfx)
|
||
|
}</span>
|
||
|
|
||
|
func(tdb *ThreadGdbmDb) SetSession(sessionId string) <span class="cov0" title="0">{
|
||
|
tdb.reserve()
|
||
|
tdb.db.SetSession(sessionId)
|
||
|
}</span>
|
||
|
|
||
|
func(tdb *ThreadGdbmDb) SetLanguage(lng *lang.Language) <span class="cov0" title="0">{
|
||
|
tdb.reserve()
|
||
|
tdb.db.SetLanguage(lng)
|
||
|
}</span>
|
||
|
|
||
|
func(tdb *ThreadGdbmDb) Safe() bool <span class="cov0" title="0">{
|
||
|
tdb.reserve()
|
||
|
v := tdb.db.Safe()
|
||
|
tdb.release()
|
||
|
return v
|
||
|
}</span>
|
||
|
|
||
|
func(tdb *ThreadGdbmDb) Prefix() uint8 <span class="cov0" title="0">{
|
||
|
tdb.reserve()
|
||
|
v := tdb.db.Prefix()
|
||
|
tdb.release()
|
||
|
return v
|
||
|
}</span>
|
||
|
|
||
|
func(tdb *ThreadGdbmDb) SetLock(typ uint8, locked bool) error <span class="cov0" title="0">{
|
||
|
tdb.reserve()
|
||
|
err := tdb.db.SetLock(typ, locked)
|
||
|
tdb.release()
|
||
|
return err
|
||
|
}</span>
|
||
|
|
||
|
func(tdb *ThreadGdbmDb) Put(ctx context.Context, key []byte, val []byte) error <span class="cov0" title="0">{
|
||
|
tdb.reserve()
|
||
|
err := tdb.db.Put(ctx, key, val)
|
||
|
tdb.release()
|
||
|
return err
|
||
|
}</span>
|
||
|
|
||
|
func(tdb *ThreadGdbmDb) Get(ctx context.Context, key []byte) ([]byte, error) <span class="cov0" title="0">{
|
||
|
tdb.reserve()
|
||
|
v, err := tdb.db.Get(ctx, key)
|
||
|
tdb.release()
|
||
|
return v, err
|
||
|
}</span>
|
||
|
|
||
|
func(tdb *ThreadGdbmDb) Close() error <span class="cov0" title="0">{
|
||
|
tdb.reserve()
|
||
|
close(dbC[tdb.connStr])
|
||
|
err := tdb.db.Close()
|
||
|
tdb.db = nil
|
||
|
return err
|
||
|
}</span>
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file18" style="display: none">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 <span class="cov0" title="0">{
|
||
|
pe := persist.NewPersister(stateStore)
|
||
|
pe = pe.WithFlush()
|
||
|
return &SimpleStorageProvider{
|
||
|
Storage: &Storage{
|
||
|
Persister: pe,
|
||
|
UserdataDb: userdataStore,
|
||
|
},
|
||
|
}
|
||
|
}</span>
|
||
|
|
||
|
func (p *SimpleStorageProvider) Get(sessionId string) (*Storage, error) <span class="cov0" title="0">{
|
||
|
return p.Storage, nil
|
||
|
}</span>
|
||
|
|
||
|
func (p *SimpleStorageProvider) Put(sessionId string, storage *Storage) error <span class="cov0" title="0">{
|
||
|
return nil
|
||
|
}</span>
|
||
|
|
||
|
func (p *SimpleStorageProvider) Close() error <span class="cov0" title="0">{
|
||
|
return p.Storage.UserdataDb.Close()
|
||
|
}</span>
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file19" style="display: none">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 <span class="cov0" title="0">{
|
||
|
return &MenuStorageService{
|
||
|
dbDir: dbDir,
|
||
|
resourceDir: resourceDir,
|
||
|
}
|
||
|
}</span>
|
||
|
|
||
|
func (ms *MenuStorageService) GetPersister(ctx context.Context) (*persist.Persister, error) <span class="cov0" title="0">{
|
||
|
ms.stateStore = NewThreadGdbmDb()
|
||
|
storeFile := path.Join(ms.dbDir, "state.gdbm")
|
||
|
err := ms.stateStore.Connect(ctx, storeFile)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">pr := persist.NewPersister(ms.stateStore)
|
||
|
logg.TraceCtxf(ctx, "menu storage service", "persist", pr, "store", ms.stateStore)
|
||
|
return pr, nil</span>
|
||
|
}
|
||
|
|
||
|
func (ms *MenuStorageService) GetUserdataDb(ctx context.Context) (db.Db, error) <span class="cov0" title="0">{
|
||
|
ms.userDataStore = NewThreadGdbmDb()
|
||
|
storeFile := path.Join(ms.dbDir, "userdata.gdbm")
|
||
|
err := ms.userDataStore.Connect(ctx, storeFile)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">return ms.userDataStore, nil</span>
|
||
|
}
|
||
|
|
||
|
func (ms *MenuStorageService) GetResource(ctx context.Context) (resource.Resource, error) <span class="cov0" title="0">{
|
||
|
ms.resourceStore = fsdb.NewFsDb()
|
||
|
err := ms.resourceStore.Connect(ctx, ms.resourceDir)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">rfs := resource.NewDbResource(ms.resourceStore)
|
||
|
return rfs, nil</span>
|
||
|
}
|
||
|
|
||
|
func (ms *MenuStorageService) GetStateStore(ctx context.Context) (db.Db, error) <span class="cov0" title="0">{
|
||
|
if ms.stateStore != nil </span><span class="cov0" title="0">{
|
||
|
panic("set up store when already exists")</span>
|
||
|
}
|
||
|
<span class="cov0" title="0">ms.stateStore = NewThreadGdbmDb()
|
||
|
storeFile := path.Join(ms.dbDir, "state.gdbm")
|
||
|
err := ms.stateStore.Connect(ctx, storeFile)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return nil, err
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">return ms.stateStore, nil</span>
|
||
|
}
|
||
|
|
||
|
func (ms *MenuStorageService) EnsureDbDir() error <span class="cov0" title="0">{
|
||
|
err := os.MkdirAll(ms.dbDir, 0700)
|
||
|
if err != nil </span><span class="cov0" title="0">{
|
||
|
return fmt.Errorf("state dir create exited with error: %v\n", err)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">return nil</span>
|
||
|
}
|
||
|
|
||
|
func (ms *MenuStorageService) Close() error <span class="cov0" title="0">{
|
||
|
errA := ms.stateStore.Close()
|
||
|
errB := ms.userDataStore.Close()
|
||
|
errC := ms.resourceStore.Close()
|
||
|
if errA != nil || errB != nil || errC != nil </span><span class="cov0" title="0">{
|
||
|
return fmt.Errorf("%v %v %v", errA, errB, errC)
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">return nil</span>
|
||
|
}
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file20" style="display: none">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 <span class="cov0" title="0">{
|
||
|
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) </span><span class="cov0" title="0">{
|
||
|
return 0
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">age := ty - by
|
||
|
anniversary := birthdate.AddDate(age, 0, 0)
|
||
|
if anniversary.After(today) </span><span class="cov0" title="0">{
|
||
|
age--
|
||
|
}</span>
|
||
|
<span class="cov0" title="0">return age</span>
|
||
|
}
|
||
|
|
||
|
// 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 <span class="cov8" title="1">{
|
||
|
currentYear := time.Now().Year()
|
||
|
return currentYear - yob
|
||
|
}</pre>
|
||
|
|
||
|
<pre class="file" id="file21" style="display: none">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 <span class="cov0" title="0">{
|
||
|
var b [2]byte
|
||
|
binary.BigEndian.PutUint16(b[:], uint16(typ))
|
||
|
return b[:]
|
||
|
}</span>
|
||
|
|
||
|
func PackKey(typ DataTyp, data []byte) []byte <span class="cov0" title="0">{
|
||
|
v := typToBytes(typ)
|
||
|
return append(v, data...)
|
||
|
}</span>
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file22" style="display: none">package utils
|
||
|
|
||
|
var isoCodes = map[string]bool{
|
||
|
"eng": true, // English
|
||
|
"swa": true, // Swahili
|
||
|
|
||
|
}
|
||
|
|
||
|
func IsValidISO639(code string) bool <span class="cov8" title="1">{
|
||
|
return isoCodes[code]
|
||
|
}</span>
|
||
|
</pre>
|
||
|
|
||
|
<pre class="file" id="file23" style="display: none">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) <span class="cov0" title="0">{
|
||
|
store.SetPrefix(db.DATATYPE_USERDATA)
|
||
|
store.SetSession(sessionId)
|
||
|
k := PackKey(typ, []byte(sessionId))
|
||
|
return store.Get(ctx, k)
|
||
|
}</span>
|
||
|
|
||
|
func (store *UserDataStore) WriteEntry(ctx context.Context, sessionId string, typ DataTyp, value []byte) error <span class="cov0" title="0">{
|
||
|
store.SetPrefix(db.DATATYPE_USERDATA)
|
||
|
store.SetSession(sessionId)
|
||
|
k := PackKey(typ, []byte(sessionId))
|
||
|
return store.Put(ctx, k, value)
|
||
|
}</span>
|
||
|
</pre>
|
||
|
|
||
|
</div>
|
||
|
</body>
|
||
|
<script>
|
||
|
(function() {
|
||
|
var files = document.getElementById('files');
|
||
|
var visible;
|
||
|
files.addEventListener('change', onChange, false);
|
||
|
function select(part) {
|
||
|
if (visible)
|
||
|
visible.style.display = 'none';
|
||
|
visible = document.getElementById(part);
|
||
|
if (!visible)
|
||
|
return;
|
||
|
files.value = part;
|
||
|
visible.style.display = 'block';
|
||
|
location.hash = part;
|
||
|
}
|
||
|
function onChange() {
|
||
|
select(files.value);
|
||
|
window.scrollTo(0, 0);
|
||
|
}
|
||
|
if (location.hash != "") {
|
||
|
select(location.hash.substr(1));
|
||
|
}
|
||
|
if (!visible) {
|
||
|
select("file0");
|
||
|
}
|
||
|
})();
|
||
|
</script>
|
||
|
</html>
|