visedriver/cmd/main.go

656 lines
15 KiB
Go
Raw Normal View History

2024-08-15 10:12:58 +02:00
package main
import (
"context"
"encoding/json"
2024-08-15 10:12:58 +02:00
"flag"
"fmt"
"io"
"net/http"
2024-08-15 10:12:58 +02:00
"os"
"path"
"time"
2024-08-15 10:12:58 +02:00
2024-08-15 12:49:36 +02:00
"git.defalsify.org/vise.git/cache"
"git.defalsify.org/vise.git/engine"
"git.defalsify.org/vise.git/persist"
2024-08-15 10:12:58 +02:00
"git.defalsify.org/vise.git/resource"
"git.defalsify.org/vise.git/state"
)
const (
USERFLAG_LANGUAGE_SET = iota + state.FLAG_USERSTART
USERFLAG_ACCOUNT_CREATED
USERFLAG_ACCOUNT_PENDING
USERFLAG_ACCOUNT_SUCCESS
2024-08-20 15:55:43 +02:00
USERFLAG_ACCOUNT_UNLOCKED
2024-08-21 19:30:59 +02:00
USERFLAG_INVALID_RECIPIENT
USERFLAG_INVALID_RECIPIENT_WITH_INVITE
2024-08-21 20:37:29 +02:00
USERFLAG_INCORRECTPIN
2024-08-22 14:44:28 +02:00
USERFLAG_UNLOCKFORUPDATE
USERFLAG_INVALID_AMOUNT
2024-08-15 10:12:58 +02:00
)
const (
createAccountURL = "https://custodial.sarafu.africa/api/account/create"
trackStatusURL = "https://custodial.sarafu.africa/api/track/"
2024-08-19 12:36:17 +02:00
checkBalanceURL = "https://custodial.sarafu.africa/api/account/status/"
)
type accountResponse struct {
Ok bool `json:"ok"`
Result struct {
CustodialId json.Number `json:"custodialId"`
PublicKey string `json:"publicKey"`
TrackingId string `json:"trackingId"`
} `json:"result"`
}
type trackStatusResponse struct {
Ok bool `json:"ok"`
Result struct {
Transaction struct {
CreatedAt time.Time `json:"createdAt"`
Status string `json:"status"`
TransferValue json.Number `json:"transferValue"`
TxHash string `json:"txHash"`
TxType string `json:"txType"`
}
} `json:"result"`
}
2024-08-19 12:36:17 +02:00
type balanceResponse struct {
Ok bool `json:"ok"`
Result struct {
Balance string `json:"balance"`
Nonce json.Number `json:"nonce"`
2024-08-19 12:36:17 +02:00
} `json:"result"`
}
2024-08-15 10:12:58 +02:00
type fsData struct {
2024-08-15 12:49:36 +02:00
path string
2024-08-21 20:37:29 +02:00
st *state.State
2024-08-15 10:12:58 +02:00
}
func (fsd *fsData) SetLanguageSelected(ctx context.Context, sym string, input []byte) (resource.Result, error) {
inputStr := string(input)
res := resource.Result{}
switch inputStr {
case "0":
res.FlagSet = []uint32{state.FLAG_LANG}
res.Content = "eng"
case "1":
res.FlagSet = []uint32{state.FLAG_LANG}
res.Content = "swa"
default:
}
res.FlagSet = append(res.FlagSet, USERFLAG_LANGUAGE_SET)
return res, nil
}
func (fsd *fsData) create_account(ctx context.Context, sym string, input []byte) (resource.Result, error) {
res := resource.Result{}
fp := fsd.path + "_data"
f, err := os.OpenFile(fp, os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644)
if err != nil {
return res, err
}
f.Close()
2024-08-21 20:49:27 +02:00
accountResp, err := createAccount()
if err != nil {
fmt.Println("Failed to create account:", err)
return res, err
}
accountData := map[string]string{
"TrackingId": accountResp.Result.TrackingId,
"PublicKey": accountResp.Result.PublicKey,
"CustodialId": accountResp.Result.CustodialId.String(),
"Status": "PENDING",
}
jsonData, err := json.Marshal(accountData)
if err != nil {
return res, err
}
err = os.WriteFile(fp, jsonData, 0644)
if err != nil {
return res, err
}
2024-08-21 20:37:29 +02:00
res.FlagSet = append(res.FlagSet, USERFLAG_ACCOUNT_CREATED)
return res, err
}
func (fsd *fsData) checkIdentifier(ctx context.Context, sym string, input []byte) (resource.Result, error) {
res := resource.Result{}
fp := fsd.path + "_data"
jsonData, err := os.ReadFile(fp)
if err != nil {
return res, err
}
var accountData map[string]string
err = json.Unmarshal(jsonData, &accountData)
if err != nil {
return res, err
}
res.Content = accountData["PublicKey"]
return res, nil
}
2024-08-21 20:37:29 +02:00
func (fsd *fsData) unLock(ctx context.Context, sym string, input []byte) (resource.Result, error) {
res := resource.Result{}
pin := string(input)
if len(input) > 0 {
if pin == "0000" {
res.FlagSet = append(res.FlagSet, USERFLAG_INCORRECTPIN)
res.FlagReset = append(res.FlagReset, USERFLAG_ACCOUNT_UNLOCKED)
return res, nil
}
if fsd.st.MatchFlag(USERFLAG_ACCOUNT_UNLOCKED, false) {
res.FlagSet = append(res.FlagSet, USERFLAG_ACCOUNT_UNLOCKED)
} else {
res.FlagReset = append(res.FlagReset, USERFLAG_ACCOUNT_UNLOCKED)
}
}
return res, nil
}
2024-08-20 15:55:43 +02:00
2024-08-21 20:37:29 +02:00
func (fsd *fsData) ResetIncorrectPin(ctx context.Context, sym string, input []byte) (resource.Result, error) {
2024-08-20 15:55:43 +02:00
res := resource.Result{}
2024-08-21 20:37:29 +02:00
isIncorrectPinSet := fsd.st.MatchFlag(USERFLAG_INCORRECTPIN, true)
if isIncorrectPinSet {
res.FlagReset = append(res.FlagReset, USERFLAG_INCORRECTPIN)
2024-08-20 15:55:43 +02:00
} else {
2024-08-21 20:37:29 +02:00
res.FlagReset = append(res.FlagReset, USERFLAG_INCORRECTPIN)
2024-08-20 15:55:43 +02:00
}
2024-08-21 20:37:29 +02:00
return res, nil
2024-08-20 15:55:43 +02:00
}
func (fsd *fsData) check_account_status(ctx context.Context, sym string, input []byte) (resource.Result, error) {
res := resource.Result{}
fp := fsd.path + "_data"
jsonData, err := os.ReadFile(fp)
if err != nil {
return res, err
}
var accountData map[string]string
err = json.Unmarshal(jsonData, &accountData)
if err != nil {
return res, err
}
status, err := checkAccountStatus(accountData["TrackingId"])
if err != nil {
fmt.Println("Error checking account status:", err)
return res, nil
}
accountData["Status"] = status
2024-08-21 20:49:27 +02:00
if status == "SUCCESS" {
2024-08-20 15:55:43 +02:00
res.FlagSet = append(res.FlagSet, USERFLAG_ACCOUNT_SUCCESS)
2024-08-21 20:37:29 +02:00
res.FlagReset = append(res.FlagReset, USERFLAG_ACCOUNT_PENDING)
} else {
2024-08-20 15:55:43 +02:00
res.FlagReset = append(res.FlagSet, USERFLAG_ACCOUNT_SUCCESS)
2024-08-21 20:37:29 +02:00
res.FlagSet = append(res.FlagReset, USERFLAG_ACCOUNT_PENDING)
}
updatedJsonData, err := json.Marshal(accountData)
if err != nil {
return res, err
}
err = os.WriteFile(fp, updatedJsonData, 0644)
if err != nil {
return res, err
}
2024-08-15 10:12:58 +02:00
return res, nil
}
func createAccount() (*accountResponse, error) {
resp, err := http.Post(createAccountURL, "application/json", nil)
if err != nil {
return nil, err
}
defer resp.Body.Close()
body, err := io.ReadAll(resp.Body)
if err != nil {
return nil, err
}
var accountResp accountResponse
err = json.Unmarshal(body, &accountResp)
if err != nil {
return nil, err
}
return &accountResp, nil
}
func checkAccountStatus(trackingId string) (string, error) {
resp, err := http.Get(trackStatusURL + trackingId)
if err != nil {
return "", err
}
defer resp.Body.Close()
body, err := io.ReadAll(resp.Body)
if err != nil {
return "", err
}
var trackResp trackStatusResponse
err = json.Unmarshal(body, &trackResp)
if err != nil {
return "", err
}
status := trackResp.Result.Transaction.Status
return status, nil
}
2024-08-21 20:37:29 +02:00
func (fsd *fsData) quit(ctx context.Context, sym string, input []byte) (resource.Result, error) {
2024-08-20 15:55:43 +02:00
res := resource.Result{
2024-08-21 20:37:29 +02:00
Content: "Your account is being created",
2024-08-20 15:55:43 +02:00
}
2024-08-21 20:37:29 +02:00
res.FlagReset = append(res.FlagReset, USERFLAG_ACCOUNT_UNLOCKED)
2024-08-20 15:55:43 +02:00
return res, nil
}
2024-08-19 12:36:17 +02:00
func (fsd *fsData) checkBalance(ctx context.Context, sym string, input []byte) (resource.Result, error) {
res := resource.Result{}
fp := fsd.path + "_data"
jsonData, err := os.ReadFile(fp)
if err != nil {
return res, err
}
var accountData map[string]string
err = json.Unmarshal(jsonData, &accountData)
if err != nil {
return res, err
}
resp, err := http.Get(checkBalanceURL + accountData["PublicKey"])
if err != nil {
return res, nil
}
defer resp.Body.Close()
body, err := io.ReadAll(resp.Body)
if err != nil {
return res, nil
}
var balanceResp balanceResponse
err = json.Unmarshal(body, &balanceResp)
if err != nil {
return res, nil
}
balance := balanceResp.Result.Balance
res.Content = balance
return res, nil
}
func (fsd *fsData) validate_recipient(ctx context.Context, sym string, input []byte) (resource.Result, error) {
2024-08-20 09:15:33 +02:00
res := resource.Result{}
recipient := string(input)
2024-08-20 14:09:02 +02:00
fp := fsd.path + "_data"
jsonData, err := os.ReadFile(fp)
if err != nil {
return res, err
}
var accountData map[string]string
err = json.Unmarshal(jsonData, &accountData)
if err != nil {
return res, err
}
2024-08-21 19:30:59 +02:00
if recipient != "0" {
// mimic invalid number check
if recipient == "000" {
res.FlagSet = append(res.FlagSet, USERFLAG_INVALID_RECIPIENT)
res.Content = recipient
2024-08-20 14:09:02 +02:00
2024-08-21 19:30:59 +02:00
return res, nil
}
2024-08-20 14:09:02 +02:00
2024-08-21 19:30:59 +02:00
accountData["Recipient"] = recipient
2024-08-20 14:09:02 +02:00
2024-08-21 19:30:59 +02:00
updatedJsonData, err := json.Marshal(accountData)
if err != nil {
return res, err
}
2024-08-20 14:09:02 +02:00
2024-08-21 19:30:59 +02:00
err = os.WriteFile(fp, updatedJsonData, 0644)
if err != nil {
return res, err
}
2024-08-20 14:09:02 +02:00
}
2024-08-20 09:15:33 +02:00
return res, nil
}
func (fsd *fsData) transaction_reset(ctx context.Context, sym string, input []byte) (resource.Result, error) {
2024-08-20 09:15:33 +02:00
res := resource.Result{}
2024-08-20 14:09:02 +02:00
fp := fsd.path + "_data"
jsonData, err := os.ReadFile(fp)
if err != nil {
return res, err
}
var accountData map[string]string
err = json.Unmarshal(jsonData, &accountData)
if err != nil {
return res, err
}
// reset the recipient
accountData["Recipient"] = ""
updatedJsonData, err := json.Marshal(accountData)
if err != nil {
return res, err
}
err = os.WriteFile(fp, updatedJsonData, 0644)
if err != nil {
return res, err
}
2024-08-21 19:30:59 +02:00
res.FlagReset = append(res.FlagReset, USERFLAG_INVALID_RECIPIENT, USERFLAG_INVALID_RECIPIENT_WITH_INVITE)
2024-08-20 14:09:02 +02:00
return res, nil
}
2024-08-22 14:44:28 +02:00
func (fsd *fsData) reset_transaction_amount(ctx context.Context, sym string, input []byte) (resource.Result, error) {
res := resource.Result{}
fp := fsd.path + "_data"
jsonData, err := os.ReadFile(fp)
if err != nil {
return res, err
}
var accountData map[string]string
err = json.Unmarshal(jsonData, &accountData)
if err != nil {
return res, err
}
// reset the amount
accountData["Amount"] = ""
updatedJsonData, err := json.Marshal(accountData)
if err != nil {
return res, err
}
err = os.WriteFile(fp, updatedJsonData, 0644)
if err != nil {
return res, err
}
res.FlagReset = append(res.FlagReset, USERFLAG_INVALID_AMOUNT)
return res, nil
}
func (fsd *fsData) max_amount(ctx context.Context, sym string, input []byte) (resource.Result, error) {
2024-08-20 14:09:02 +02:00
res := resource.Result{}
// mimic a max amount
res.Content = "10.00"
return res, nil
}
func (fsd *fsData) validate_amount(ctx context.Context, sym string, input []byte) (resource.Result, error) {
res := resource.Result{}
amount := string(input)
fp := fsd.path + "_data"
jsonData, err := os.ReadFile(fp)
if err != nil {
return res, err
}
var accountData map[string]string
err = json.Unmarshal(jsonData, &accountData)
if err != nil {
return res, err
}
2024-08-22 14:44:28 +02:00
if amount != "0" {
// mimic invalid amount
if amount == "00" {
res.FlagSet = append(res.FlagSet, USERFLAG_INVALID_AMOUNT)
res.Content = amount
2024-08-22 14:44:28 +02:00
return res, nil
}
2024-08-22 14:44:28 +02:00
res.Content = amount
2024-08-22 14:44:28 +02:00
accountData["Amount"] = amount
2024-08-22 14:44:28 +02:00
updatedJsonData, err := json.Marshal(accountData)
if err != nil {
return res, err
}
2024-08-22 14:44:28 +02:00
err = os.WriteFile(fp, updatedJsonData, 0644)
if err != nil {
return res, err
}
return res, nil
}
return res, nil
}
func (fsd *fsData) get_recipient(ctx context.Context, sym string, input []byte) (resource.Result, error) {
res := resource.Result{}
fp := fsd.path + "_data"
jsonData, err := os.ReadFile(fp)
if err != nil {
return res, err
}
var accountData map[string]string
err = json.Unmarshal(jsonData, &accountData)
if err != nil {
return res, err
}
res.Content = accountData["Recipient"]
return res, nil
}
func (fsd *fsData) get_sender(ctx context.Context, sym string, input []byte) (resource.Result, error) {
res := resource.Result{}
fp := fsd.path + "_data"
jsonData, err := os.ReadFile(fp)
if err != nil {
return res, err
}
var accountData map[string]string
err = json.Unmarshal(jsonData, &accountData)
if err != nil {
return res, err
}
res.Content = accountData["PublicKey"]
2024-08-20 09:15:33 +02:00
return res, nil
}
2024-08-21 20:37:29 +02:00
func (fsd *fsData) quitWithBalance(ctx context.Context, sym string, input []byte) (resource.Result, error) {
2024-08-21 20:49:27 +02:00
res := resource.Result{}
2024-08-21 20:37:29 +02:00
fp := fsd.path + "_data"
jsonData, err := os.ReadFile(fp)
if err != nil {
return res, err
}
var accountData map[string]string
err = json.Unmarshal(jsonData, &accountData)
if err != nil {
return res, err
}
resp, err := http.Get(checkBalanceURL + accountData["PublicKey"])
if err != nil {
return res, nil
}
defer resp.Body.Close()
body, err := io.ReadAll(resp.Body)
if err != nil {
return res, nil
}
var balanceResp balanceResponse
err = json.Unmarshal(body, &balanceResp)
if err != nil {
return res, nil
}
balance := balanceResp.Result.Balance
2024-08-21 20:49:27 +02:00
res.Content = fmt.Sprintf("Your account balance is: %s", balance)
2024-08-21 20:37:29 +02:00
res.FlagReset = append(res.FlagReset, USERFLAG_ACCOUNT_UNLOCKED)
return res, nil
}
2024-08-15 12:49:36 +02:00
var (
scriptDir = path.Join("services", "registration")
)
2024-08-15 10:12:58 +02:00
func main() {
var dir string
var root string
var size uint
var sessionId string
flag.StringVar(&dir, "d", ".", "resource dir to read from")
flag.UintVar(&size, "s", 0, "max size of output")
2024-08-16 16:57:36 +02:00
flag.StringVar(&root, "root", "root", "entry point symbol")
2024-08-15 10:12:58 +02:00
flag.StringVar(&sessionId, "session-id", "default", "session id")
flag.Parse()
fmt.Fprintf(os.Stderr, "starting session at symbol '%s' using resource dir: %s\n", root, dir)
ctx := context.Background()
2024-08-22 14:44:28 +02:00
st := state.NewState(15)
st.UseDebug()
2024-08-19 09:59:20 +02:00
state.FlagDebugger.Register(USERFLAG_LANGUAGE_SET, "LANGUAGE_CHANGE")
state.FlagDebugger.Register(USERFLAG_ACCOUNT_CREATED, "ACCOUNT_CREATED")
state.FlagDebugger.Register(USERFLAG_ACCOUNT_SUCCESS, "ACCOUNT_SUCCESS")
state.FlagDebugger.Register(USERFLAG_ACCOUNT_PENDING, "ACCOUNT_PENDING")
2024-08-21 20:37:29 +02:00
state.FlagDebugger.Register(USERFLAG_INCORRECTPIN, "INCORRECTPIN")
2024-08-19 09:59:20 +02:00
2024-08-15 12:49:36 +02:00
rfs := resource.NewFsResource(scriptDir)
ca := cache.NewCache()
cfg := engine.Config{
2024-08-16 16:57:36 +02:00
Root: "root",
2024-08-15 12:49:36 +02:00
SessionId: sessionId,
}
2024-08-15 10:12:58 +02:00
2024-08-15 12:49:36 +02:00
dp := path.Join(scriptDir, ".state")
err := os.MkdirAll(dp, 0700)
2024-08-15 10:12:58 +02:00
if err != nil {
2024-08-15 12:49:36 +02:00
fmt.Fprintf(os.Stderr, "state dir create exited with error: %v\n", err)
2024-08-15 10:12:58 +02:00
os.Exit(1)
}
2024-08-15 12:49:36 +02:00
pr := persist.NewFsPersister(dp)
en, err := engine.NewPersistedEngine(ctx, cfg, pr, rfs)
2024-08-21 20:37:29 +02:00
2024-08-15 12:49:36 +02:00
if err != nil {
pr = pr.WithContent(&st, ca)
err = pr.Save(cfg.SessionId)
if err != nil {
fmt.Fprintf(os.Stderr, "Failed to save state with error: %v\n", err)
}
2024-08-15 12:49:36 +02:00
en, err = engine.NewPersistedEngine(ctx, cfg, pr, rfs)
if err != nil {
fmt.Fprintf(os.Stderr, "engine create exited with error: %v\n", err)
os.Exit(1)
}
}
fp := path.Join(dp, sessionId)
fs := &fsData{
path: fp,
2024-08-21 20:37:29 +02:00
st: &st,
2024-08-15 12:49:36 +02:00
}
rfs.AddLocalFunc("select_language", fs.SetLanguageSelected)
rfs.AddLocalFunc("create_account", fs.create_account)
rfs.AddLocalFunc("check_identifier", fs.checkIdentifier)
rfs.AddLocalFunc("check_account_status", fs.check_account_status)
2024-08-21 20:37:29 +02:00
rfs.AddLocalFunc("unlock_account", fs.unLock)
rfs.AddLocalFunc("quit", fs.quit)
2024-08-19 12:36:17 +02:00
rfs.AddLocalFunc("check_balance", fs.checkBalance)
2024-08-20 14:09:02 +02:00
rfs.AddLocalFunc("validate_recipient", fs.validate_recipient)
2024-08-20 09:15:33 +02:00
rfs.AddLocalFunc("transaction_reset", fs.transaction_reset)
2024-08-20 14:09:02 +02:00
rfs.AddLocalFunc("max_amount", fs.max_amount)
rfs.AddLocalFunc("validate_amount", fs.validate_amount)
2024-08-22 14:44:28 +02:00
rfs.AddLocalFunc("reset_transaction_amount", fs.reset_transaction_amount)
rfs.AddLocalFunc("get_recipient", fs.get_recipient)
rfs.AddLocalFunc("get_sender", fs.get_sender)
2024-08-21 20:37:29 +02:00
rfs.AddLocalFunc("reset_incorrect", fs.ResetIncorrectPin)
2024-08-21 20:49:27 +02:00
rfs.AddLocalFunc("quit_with_balance", fs.quitWithBalance)
2024-08-15 10:12:58 +02:00
cont, err := en.Init(ctx)
2024-08-21 20:37:29 +02:00
en.SetDebugger(engine.NewSimpleDebug(nil))
2024-08-15 10:12:58 +02:00
if err != nil {
fmt.Fprintf(os.Stderr, "engine init exited with error: %v\n", err)
os.Exit(1)
}
if !cont {
_, err = en.WriteResult(ctx, os.Stdout)
if err != nil {
fmt.Fprintf(os.Stderr, "dead init write error: %v\n", err)
os.Exit(1)
}
2024-08-15 12:49:36 +02:00
err = en.Finish()
if err != nil {
fmt.Fprintf(os.Stderr, "engine finish error: %v\n", err)
os.Exit(1)
}
2024-08-15 10:12:58 +02:00
os.Stdout.Write([]byte{0x0a})
os.Exit(0)
}
err = engine.Loop(ctx, en, os.Stdin, os.Stdout)
if err != nil {
fmt.Fprintf(os.Stderr, "loop exited with error: %v\n", err)
os.Exit(1)
}
}