package ussd import ( "bytes" "context" "fmt" "path" "regexp" "strconv" "strings" "git.defalsify.org/vise.git/engine" "git.defalsify.org/vise.git/lang" "git.defalsify.org/vise.git/resource" "git.defalsify.org/vise.git/state" "git.grassecon.net/urdt/ussd/internal/handlers/server" "git.grassecon.net/urdt/ussd/internal/models" "git.grassecon.net/urdt/ussd/internal/utils" "gopkg.in/leonelquinteros/gotext.v1" ) var ( scriptDir = path.Join( "services", "registration") translationDir = path.Join(scriptDir, "locale") ) type FSData struct { Path string St *state.State } type Handlers struct { fs *FSData accountFileHandler *utils.AccountFileHandler } func NewHandlers(path string, st *state.State) *Handlers { return &Handlers{ fs: &FSData{ Path: path, St: st, }, accountFileHandler: utils.NewAccountFileHandler(path + "_data"), } } func (h *Handlers) SetLanguage(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, models.USERFLAG_LANGUAGE_SET) return res, nil } func (h *Handlers) CreateAccount(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} err := h.accountFileHandler.EnsureFileExists() if err != nil { return res, err } // if an account exists, set the flag and return existingAccountData, err := h.accountFileHandler.ReadAccountData() if existingAccountData != nil { res.FlagSet = append(res.FlagSet, models.USERFLAG_ACCOUNT_CREATED) return res, err } accountResp, err := server.CreateAccount() if err != nil { res.FlagSet = append(res.FlagSet, models.USERFLAG_ACCOUNT_CREATION_FAILED) return res, err } accountData := map[string]string{ "TrackingId": accountResp.Result.TrackingId, "PublicKey": accountResp.Result.PublicKey, "CustodialId": accountResp.Result.CustodialId.String(), "Status": "PENDING", "Gender": "Not provided", "YOB": "Not provided", "Location": "Not provided", "Offerings": "Not provided", "FirstName": "Not provided", "FamilyName": "Not provided", } err = h.accountFileHandler.WriteAccountData(accountData) if err != nil { return res, err } res.FlagSet = append(res.FlagSet, models.USERFLAG_ACCOUNT_CREATED) return res, err } // SavePin persists the user's PIN choice into the filesystem func (h *Handlers) SavePin(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountPIN := string(input) accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } // Validate that the PIN is a 4-digit number if !isValidPIN(accountPIN) { res.FlagSet = append(res.FlagSet, models.USERFLAG_INCORRECTPIN) return res, nil } res.FlagReset = append(res.FlagReset, models.USERFLAG_INCORRECTPIN) accountData["AccountPIN"] = accountPIN err = h.accountFileHandler.WriteAccountData(accountData) if err != nil { return res, err } return res, nil } func (h *Handlers) SetResetSingleEdit(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} menuOption := string(input) switch menuOption { case "2": res.FlagReset = append(res.FlagSet, models.USERFLAG_UNLOCKFORUPDATE) res.FlagSet = append(res.FlagSet, models.USERFLAG_SINGLE_EDIT) case "3": res.FlagReset = append(res.FlagSet, models.USERFLAG_UNLOCKFORUPDATE) res.FlagSet = append(res.FlagSet, models.USERFLAG_SINGLE_EDIT) case "4": res.FlagReset = append(res.FlagSet, models.USERFLAG_UNLOCKFORUPDATE) res.FlagSet = append(res.FlagSet, models.USERFLAG_SINGLE_EDIT) default: res.FlagReset = append(res.FlagReset, models.USERFLAG_SINGLE_EDIT) } return res, nil } func (h *Handlers) VerifyPin(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } if bytes.Equal(input, []byte(accountData["AccountPIN"])) { res.FlagSet = []uint32{models.USERFLAG_VALIDPIN} res.FlagReset = []uint32{models.USERFLAG_PINMISMATCH} res.FlagSet = append(res.FlagSet, models.USERFLAG_PIN_SET) } else { res.FlagSet = []uint32{models.USERFLAG_PINMISMATCH} } return res, nil } func isValidPIN(pin string) bool { match, _ := regexp.MatchString(`^\d{4}$`, pin) return match } func codeFromCtx(ctx context.Context) string { var code string engine.Logg.DebugCtxf(ctx, "in msg", "ctx", ctx, "val", code) if ctx.Value("Language") != nil { lang := ctx.Value("Language").(lang.Language) code = lang.Code } return code } // SaveFirstname updates the first name in a JSON data file with the provided input. func (h *Handlers) SaveFirstname(cxt context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } if len(input) > 0 { name := string(input) accountData["FirstName"] = name err = h.accountFileHandler.WriteAccountData(accountData) if err != nil { return res, err } } return res, nil } // SaveFamilyname updates the family name in a JSON data file with the provided input. func (h *Handlers) SaveFamilyname(cxt context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } if len(input) > 0 { secondname := string(input) accountData["FamilyName"] = secondname err = h.accountFileHandler.WriteAccountData(accountData) if err != nil { return res, err } } return res, nil } // SaveYOB updates the Year of Birth(YOB) in a JSON data file with the provided input. func (h *Handlers) SaveYob(cxt context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } yob := string(input) if len(yob) == 4 { yob := string(input) accountData["YOB"] = yob err = h.accountFileHandler.WriteAccountData(accountData) if err != nil { return res, err } } return res, nil } // SaveLocation updates the location in a JSON data file with the provided input. func (h *Handlers) SaveLocation(cxt context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } if len(input) > 0 { location := string(input) accountData["Location"] = location err = h.accountFileHandler.WriteAccountData(accountData) if err != nil { return res, err } } return res, nil } // SaveGender updates the gender in a JSON data file with the provided input. func (h *Handlers) SaveGender(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } if len(input) > 0 { gender := string(input) switch gender { case "1": gender = "Male" case "2": gender = "Female" case "3": gender = "Other" } accountData["Gender"] = gender err = h.accountFileHandler.WriteAccountData(accountData) if err != nil { return res, err } } return res, nil } // SaveOfferings updates the offerings(goods and services provided by the user) in a JSON data file with the provided input. func (h *Handlers) SaveOfferings(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } if len(input) > 0 { offerings := string(input) accountData["Offerings"] = offerings err = h.accountFileHandler.WriteAccountData(accountData) if err != nil { return res, err } } return res, nil } func (h *Handlers) ResetUnlockForUpdate(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} res.FlagReset = append(res.FlagReset, models.USERFLAG_UNLOCKFORUPDATE) return res, nil } func (h *Handlers) ResetAccountUnlocked(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} res.FlagReset = append(res.FlagReset, models.USERFLAG_ACCOUNT_UNLOCKED) return res, nil } func (h *Handlers) CheckIdentifier(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } res.Content = accountData["PublicKey"] return res, nil } // Unlock 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) Unlock(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} pin := string(input) accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } if len(input) == 4 { if pin != accountData["AccountPIN"] { res.FlagSet = append(res.FlagSet, models.USERFLAG_INCORRECTPIN) res.FlagReset = append(res.FlagReset, models.USERFLAG_ACCOUNT_UNLOCKED) return res, nil } if h.fs.St.MatchFlag(models.USERFLAG_ACCOUNT_UNLOCKED, false) { res.FlagReset = append(res.FlagReset, models.USERFLAG_INCORRECTPIN) res.FlagSet = append(res.FlagSet, models.USERFLAG_UNLOCKFORUPDATE) res.FlagSet = append(res.FlagSet, models.USERFLAG_ACCOUNT_UNLOCKED) } else { res.FlagSet = append(res.FlagSet, models.USERFLAG_UNLOCKFORUPDATE) res.FlagReset = append(res.FlagReset, models.USERFLAG_ACCOUNT_UNLOCKED) } } return res, nil } func (h *Handlers) ResetIncorrectPin(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} res.FlagReset = append(res.FlagReset, models.USERFLAG_INCORRECTPIN) return res, nil } func (h *Handlers) CheckAccountStatus(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } status, err := server.CheckAccountStatus(accountData["TrackingId"]) if err != nil { fmt.Println("Error checking account status:", err) return res, nil } accountData["Status"] = status if status == "SUCCESS" { res.FlagSet = append(res.FlagSet, models.USERFLAG_ACCOUNT_SUCCESS) res.FlagReset = append(res.FlagReset, models.USERFLAG_ACCOUNT_PENDING) } else { res.FlagReset = append(res.FlagSet, models.USERFLAG_ACCOUNT_SUCCESS) res.FlagSet = append(res.FlagReset, models.USERFLAG_ACCOUNT_PENDING) } err = h.accountFileHandler.WriteAccountData(accountData) if err != nil { return res, err } return res, nil } func (h *Handlers) Quit(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} switch codeFromCtx(ctx) { case "swa": res.Content = "Asante kwa kutumia huduma ya Sarafu. Kwaheri!" default: res.Content = "Thank you for using Sarafu. Goodbye!" } res.FlagReset = append(res.FlagReset, models.USERFLAG_ACCOUNT_UNLOCKED) return res, nil } func (h *Handlers) VerifyYob(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} date := string(input) _, err := strconv.Atoi(date) if err != nil { // If conversion fails, input is not numeric res.FlagSet = append(res.FlagSet, models.USERFLAG_INCORRECTDATEFORMAT) return res, nil } if len(date) == 4 { res.FlagReset = append(res.FlagReset, models.USERFLAG_INCORRECTDATEFORMAT) } else { res.FlagSet = append(res.FlagSet, models.USERFLAG_INCORRECTDATEFORMAT) } return res, nil } func (h *Handlers) ResetIncorrectYob(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} res.FlagReset = append(res.FlagReset, models.USERFLAG_INCORRECTDATEFORMAT) return res, nil } func (h *Handlers) CheckBalance(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } balance, err := server.CheckBalance(accountData["PublicKey"]) if err != nil { return res, nil } res.Content = balance return res, nil } func (h *Handlers) ValidateRecipient(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} recipient := string(input) accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } if recipient != "0" { // mimic invalid number check if recipient == "000" { res.FlagSet = append(res.FlagSet, models.USERFLAG_INVALID_RECIPIENT) res.Content = recipient return res, nil } accountData["Recipient"] = recipient err = h.accountFileHandler.WriteAccountData(accountData) if err != nil { return res, err } } return res, nil } func (h *Handlers) TransactionReset(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } // reset the transaction accountData["Recipient"] = "" accountData["Amount"] = "" err = h.accountFileHandler.WriteAccountData(accountData) if err != nil { return res, err } res.FlagReset = append(res.FlagReset, models.USERFLAG_INVALID_RECIPIENT, models.USERFLAG_INVALID_RECIPIENT_WITH_INVITE) return res, nil } func (h *Handlers) ResetTransactionAmount(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } // reset the amount accountData["Amount"] = "" err = h.accountFileHandler.WriteAccountData(accountData) if err != nil { return res, err } res.FlagReset = append(res.FlagReset, models.USERFLAG_INVALID_AMOUNT) return res, nil } func (h *Handlers) MaxAmount(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } balance, err := server.CheckBalance(accountData["PublicKey"]) if err != nil { return res, nil } res.Content = balance return res, nil } func (h *Handlers) ValidateAmount(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} amountStr := string(input) accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } balanceStr, err := server.CheckBalance(accountData["PublicKey"]) if err != nil { return res, err } res.Content = balanceStr // Parse the balance balanceParts := strings.Split(balanceStr, " ") if len(balanceParts) != 2 { return res, fmt.Errorf("unexpected balance format: %s", balanceStr) } balanceValue, err := strconv.ParseFloat(balanceParts[0], 64) if err != nil { return res, fmt.Errorf("failed to parse balance: %v", err) } // Extract numeric part from input re := regexp.MustCompile(`^(\d+(\.\d+)?)\s*(?:CELO)?$`) matches := re.FindStringSubmatch(strings.TrimSpace(amountStr)) if len(matches) < 2 { res.FlagSet = append(res.FlagSet, models.USERFLAG_INVALID_AMOUNT) res.Content = amountStr return res, nil } inputAmount, err := strconv.ParseFloat(matches[1], 64) if err != nil { res.FlagSet = append(res.FlagSet, models.USERFLAG_INVALID_AMOUNT) res.Content = amountStr return res, nil } if inputAmount > balanceValue { res.FlagSet = append(res.FlagSet, models.USERFLAG_INVALID_AMOUNT) res.Content = amountStr return res, nil } res.Content = fmt.Sprintf("%.3f", inputAmount) // Format to 3 decimal places accountData["Amount"] = res.Content err = h.accountFileHandler.WriteAccountData(accountData) if err != nil { return res, err } return res, nil } func (h *Handlers) GetRecipient(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } res.Content = accountData["Recipient"] return res, nil } // GetProfileInfo retrieves and formats the profile information of a user from a JSON data file. func (h *Handlers) GetProfileInfo(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} var age string accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } var name string if accountData["FirstName"] == "Not provided" || accountData["FamilyName"] == "Not provided" { name = "Not provided" } else { name = accountData["FirstName"] + " " + accountData["FamilyName"] } gender := accountData["Gender"] yob := accountData["YOB"] location := accountData["Location"] offerings := accountData["Offerings"] if yob == "Not provided" { age = "Not provided" } else { ageInt, err := strconv.Atoi(yob) if err != nil { return res, nil } age = strconv.Itoa(utils.CalculateAgeWithYOB(ageInt)) } formattedData := fmt.Sprintf("Name: %s\nGender: %s\nAge: %s\nLocation: %s\nYou provide: %s\n", name, gender, age, location, offerings) res.Content = formattedData return res, nil } // GetSender retrieves the public key from a JSON data file. func (h *Handlers) GetSender(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } res.Content = accountData["PublicKey"] return res, nil } // GetAmount retrieves the amount from a JSON data file. func (h *Handlers) GetAmount(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } res.Content = accountData["Amount"] return res, nil } // QuickWithBalance retrieves the balance for a given public key from the custodial balance API endpoint before // gracefully exiting the session. func (h *Handlers) QuitWithBalance(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} code := codeFromCtx(ctx) l := gotext.NewLocale(translationDir, code) l.AddDomain("default") accountData, err := h.accountFileHandler.ReadAccountData() if err != nil { return res, err } balance, err := server.CheckBalance(accountData["PublicKey"]) if err != nil { return res, nil } res.Content = l.Get("Your account balance is %s", balance) res.FlagReset = append(res.FlagReset, models.USERFLAG_ACCOUNT_UNLOCKED) return res, nil } func (h *Handlers) InitiateTransaction(ctx context.Context, sym string, input []byte) (resource.Result, error) { res := resource.Result{} // TODO // Use the amount, recipient and sender to call the API and initialize the transaction res.FlagReset = append(res.FlagReset, models.USERFLAG_ACCOUNT_UNLOCKED) return res, nil }