2024-08-15 10:12:58 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2024-08-16 12:38:58 +02:00
|
|
|
"encoding/json"
|
2024-08-15 10:12:58 +02:00
|
|
|
"flag"
|
|
|
|
"fmt"
|
2024-08-16 12:38:58 +02:00
|
|
|
"io"
|
|
|
|
"net/http"
|
2024-08-15 10:12:58 +02:00
|
|
|
"os"
|
|
|
|
"path"
|
2024-08-16 12:38:58 +02:00
|
|
|
"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
|
2024-08-16 12:38:58 +02:00
|
|
|
USERFLAG_ACCOUNT_CREATED
|
|
|
|
USERFLAG_ACCOUNT_PENDING
|
|
|
|
USERFLAG_ACCOUNT_SUCCESS
|
2024-08-15 10:12:58 +02:00
|
|
|
)
|
|
|
|
|
2024-08-16 12:38:58 +02:00
|
|
|
const (
|
|
|
|
createAccountURL = "https://custodial.sarafu.africa/api/account/create"
|
|
|
|
trackStatusURL = "https://custodial.sarafu.africa/api/track/"
|
|
|
|
)
|
|
|
|
|
|
|
|
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-15 10:12:58 +02:00
|
|
|
type fsData struct {
|
2024-08-15 12:49:36 +02:00
|
|
|
path string
|
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:
|
|
|
|
}
|
2024-08-16 12:38:58 +02:00
|
|
|
|
|
|
|
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()
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
|
|
res.FlagSet = []uint32{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
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
if status == "SUCCESS" {
|
|
|
|
res.FlagSet = []uint32{USERFLAG_ACCOUNT_SUCCESS}
|
|
|
|
res.FlagReset = []uint32{USERFLAG_ACCOUNT_PENDING}
|
|
|
|
} else {
|
|
|
|
res.FlagSet = []uint32{USERFLAG_ACCOUNT_PENDING}
|
|
|
|
res.FlagReset = []uint32{USERFLAG_ACCOUNT_SUCCESS}
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2024-08-16 12:38:58 +02:00
|
|
|
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-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-15 13:58:44 +02:00
|
|
|
flag.StringVar(&root, "root", "registration", "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-16 12:38:58 +02:00
|
|
|
st := state.NewState(7)
|
2024-08-15 12:49:36 +02:00
|
|
|
rfs := resource.NewFsResource(scriptDir)
|
|
|
|
ca := cache.NewCache()
|
|
|
|
cfg := engine.Config{
|
2024-08-15 13:58:44 +02:00
|
|
|
Root: "registration",
|
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)
|
|
|
|
if err != nil {
|
|
|
|
pr = pr.WithContent(&st, ca)
|
|
|
|
err = pr.Save(cfg.SessionId)
|
|
|
|
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,
|
|
|
|
}
|
|
|
|
rfs.AddLocalFunc("select_language", fs.SetLanguageSelected)
|
2024-08-16 12:38:58 +02:00
|
|
|
rfs.AddLocalFunc("create_account", fs.create_account)
|
|
|
|
rfs.AddLocalFunc("check_identifier", fs.checkIdentifier)
|
|
|
|
rfs.AddLocalFunc("check_account_status", fs.check_account_status)
|
|
|
|
|
2024-08-15 10:12:58 +02:00
|
|
|
cont, err := en.Init(ctx)
|
|
|
|
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)
|
|
|
|
}
|
|
|
|
}
|