forked from urdt/ussd
		
	Merge branch 'master' into profile-edit-traverse
This commit is contained in:
		
						commit
						91cd6077ce
					
				| @ -151,7 +151,7 @@ func GetTemporaryVoucherData(ctx context.Context, store DataStore, sessionId str | ||||
| 	return data, nil | ||||
| } | ||||
| 
 | ||||
| // UpdateVoucherData sets the active voucher data in the DataStore.
 | ||||
| // UpdateVoucherData updates the active voucher data in the DataStore.
 | ||||
| func UpdateVoucherData(ctx context.Context, store DataStore, sessionId string, data *dataserviceapi.TokenHoldings) error { | ||||
| 	logg.TraceCtxf(ctx, "dtal", "data", data) | ||||
| 	// Active voucher data entries
 | ||||
|  | ||||
| @ -55,6 +55,9 @@ func(f *BaseSessionHandler) Process(rqs RequestSession) (RequestSession, error) | ||||
| 	} | ||||
| 
 | ||||
| 	f.hn = f.hn.WithPersister(rqs.Storage.Persister) | ||||
| 	defer func() { | ||||
| 		f.hn.Exit() | ||||
| 	}() | ||||
| 	eni := f.GetEngine(rqs.Config, f.rs, rqs.Storage.Persister) | ||||
| 	en, ok := eni.(*engine.DefaultEngine) | ||||
| 	if !ok { | ||||
|  | ||||
| @ -10,7 +10,6 @@ import ( | ||||
| 	"strings" | ||||
| 
 | ||||
| 	"git.defalsify.org/vise.git/asm" | ||||
| 	"github.com/grassrootseconomics/eth-custodial/pkg/api" | ||||
| 
 | ||||
| 	"git.defalsify.org/vise.git/cache" | ||||
| 	"git.defalsify.org/vise.git/db" | ||||
| @ -26,14 +25,13 @@ import ( | ||||
| 	"gopkg.in/leonelquinteros/gotext.v1" | ||||
| 
 | ||||
| 	"git.grassecon.net/urdt/ussd/internal/storage" | ||||
| 	dataserviceapi "github.com/grassrootseconomics/ussd-data-service/pkg/api" | ||||
| ) | ||||
| 
 | ||||
| var ( | ||||
| 	logg           = logging.NewVanilla().WithDomain("ussdmenuhandler") | ||||
| 	scriptDir      = path.Join("services", "registration") | ||||
| 	translationDir = path.Join(scriptDir, "locale") | ||||
| 	okResponse     *api.OKResponse | ||||
| 	errResponse    *api.ErrResponse | ||||
| ) | ||||
| 
 | ||||
| // Define the regex patterns as constants
 | ||||
| @ -117,6 +115,9 @@ func (h *Handlers) Init(ctx context.Context, sym string, input []byte) (resource | ||||
| 		logg.WarnCtxf(ctx, "handler init called before it is ready or more than once", "state", h.st, "cache", h.ca) | ||||
| 		return r, nil | ||||
| 	} | ||||
| 	defer func() { | ||||
| 		h.Exit() | ||||
| 	}() | ||||
| 
 | ||||
| 	h.st = h.pe.GetState() | ||||
| 	h.ca = h.pe.GetMemory() | ||||
| @ -136,13 +137,16 @@ func (h *Handlers) Init(ctx context.Context, sym string, input []byte) (resource | ||||
| 		logg.ErrorCtxf(ctx, "perister fail in handler", "state", h.st, "cache", h.ca) | ||||
| 		return r, fmt.Errorf("cannot get state and memory for handler") | ||||
| 	} | ||||
| 	h.pe = nil | ||||
| 
 | ||||
| 	logg.DebugCtxf(ctx, "handler has been initialized", "state", h.st, "cache", h.ca) | ||||
| 
 | ||||
| 	return r, nil | ||||
| } | ||||
| 
 | ||||
| func (h *Handlers) Exit() { | ||||
| 	h.pe = nil | ||||
| } | ||||
| 
 | ||||
| // SetLanguage sets the language across the menu
 | ||||
| func (h *Handlers) SetLanguage(ctx context.Context, sym string, input []byte) (resource.Result, error) { | ||||
| 	var res resource.Result | ||||
| @ -151,7 +155,8 @@ func (h *Handlers) SetLanguage(ctx context.Context, sym string, input []byte) (r | ||||
| 	code := strings.Split(symbol, "_")[1] | ||||
| 
 | ||||
| 	if !utils.IsValidISO639(code) { | ||||
| 		return res, nil | ||||
| 		//Fallback to english instead?
 | ||||
| 		code = "eng" | ||||
| 	} | ||||
| 	res.FlagSet = append(res.FlagSet, state.FLAG_LANG) | ||||
| 	res.Content = code | ||||
| @ -806,12 +811,11 @@ func (h *Handlers) VerifyYob(ctx context.Context, sym string, input []byte) (res | ||||
| 		return res, nil | ||||
| 	} | ||||
| 
 | ||||
| 	if len(date) == 4 { | ||||
| 	if utils.IsValidYOb(date) { | ||||
| 		res.FlagReset = append(res.FlagReset, flag_incorrect_date_format) | ||||
| 	} else { | ||||
| 		res.FlagSet = append(res.FlagSet, flag_incorrect_date_format) | ||||
| 	} | ||||
| 
 | ||||
| 	return res, nil | ||||
| } | ||||
| 
 | ||||
| @ -860,7 +864,17 @@ func (h *Handlers) CheckBalance(ctx context.Context, sym string, input []byte) ( | ||||
| 		return res, err | ||||
| 	} | ||||
| 
 | ||||
| 	res.Content = l.Get("Balance: %s\n", fmt.Sprintf("%s %s", activeBal, activeSym)) | ||||
| 	// Convert activeBal from []byte to float64
 | ||||
| 	balFloat, err := strconv.ParseFloat(string(activeBal), 64) | ||||
| 	if err != nil { | ||||
| 		logg.ErrorCtxf(ctx, "failed to parse activeBal as float", "value", string(activeBal), "error", err) | ||||
| 		return res, err | ||||
| 	} | ||||
| 
 | ||||
| 	// Format to 2 decimal places
 | ||||
| 	balStr := fmt.Sprintf("%.2f %s", balFloat, activeSym) | ||||
| 
 | ||||
| 	res.Content = l.Get("Balance: %s\n", balStr) | ||||
| 
 | ||||
| 	return res, nil | ||||
| } | ||||
| @ -1450,14 +1464,7 @@ func (h *Handlers) GetProfileInfo(ctx context.Context, sym string, input []byte) | ||||
| 	offerings := getEntryOrDefault(store.ReadEntry(ctx, sessionId, common.DATA_OFFERINGS)) | ||||
| 
 | ||||
| 	// Construct the full name
 | ||||
| 	name := defaultValue | ||||
| 	if familyName != defaultValue { | ||||
| 		if firstName == defaultValue { | ||||
| 			name = familyName | ||||
| 		} else { | ||||
| 			name = firstName + " " + familyName | ||||
| 		} | ||||
| 	} | ||||
| 	name := utils.ConstructName(firstName, familyName, defaultValue) | ||||
| 
 | ||||
| 	// Calculate age from year of birth
 | ||||
| 	age := defaultValue | ||||
| @ -1596,6 +1603,38 @@ func (h *Handlers) CheckVouchers(ctx context.Context, sym string, input []byte) | ||||
| 		return res, nil | ||||
| 	} | ||||
| 
 | ||||
| 	// check the current active sym and update the data
 | ||||
| 	activeSym, _ := store.ReadEntry(ctx, sessionId, common.DATA_ACTIVE_SYM) | ||||
| 	if activeSym != nil { | ||||
| 		activeSymStr := string(activeSym) | ||||
| 
 | ||||
| 		// Find the matching voucher data
 | ||||
| 		var activeData *dataserviceapi.TokenHoldings | ||||
| 		for _, voucher := range vouchersResp { | ||||
| 			if voucher.TokenSymbol == activeSymStr { | ||||
| 				activeData = &voucher | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
| 
 | ||||
| 		if activeData == nil { | ||||
| 			logg.ErrorCtxf(ctx, "activeSym not found in vouchers", "activeSym", activeSymStr) | ||||
| 			return res, fmt.Errorf("activeSym %s not found in vouchers", activeSymStr) | ||||
| 		} | ||||
| 
 | ||||
| 		// Scale down the balance
 | ||||
| 		scaledBalance := common.ScaleDownBalance(activeData.Balance, activeData.TokenDecimals) | ||||
| 
 | ||||
| 		// Update the balance field with the scaled value
 | ||||
| 		activeData.Balance = scaledBalance | ||||
| 
 | ||||
| 		// Pass the matching voucher data to UpdateVoucherData
 | ||||
| 		if err := common.UpdateVoucherData(ctx, h.userdataStore, sessionId, activeData); err != nil { | ||||
| 			logg.ErrorCtxf(ctx, "failed on UpdateVoucherData", "error", err) | ||||
| 			return res, err | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	data := common.ProcessVouchers(vouchersResp) | ||||
| 
 | ||||
| 	// Store all voucher data
 | ||||
| @ -1640,6 +1679,10 @@ func (h *Handlers) ViewVoucher(ctx context.Context, sym string, input []byte) (r | ||||
| 		return res, fmt.Errorf("missing session") | ||||
| 	} | ||||
| 
 | ||||
| 	code := codeFromCtx(ctx) | ||||
| 	l := gotext.NewLocale(translationDir, code) | ||||
| 	l.AddDomain("default") | ||||
| 
 | ||||
| 	flag_incorrect_voucher, _ := h.flagManager.GetFlag("flag_incorrect_voucher") | ||||
| 
 | ||||
| 	inputStr := string(input) | ||||
| @ -1664,7 +1707,7 @@ func (h *Handlers) ViewVoucher(ctx context.Context, sym string, input []byte) (r | ||||
| 	} | ||||
| 
 | ||||
| 	res.FlagReset = append(res.FlagReset, flag_incorrect_voucher) | ||||
| 	res.Content = fmt.Sprintf("%s\n%s", metadata.TokenSymbol, metadata.Balance) | ||||
| 	res.Content = l.Get("Symbol: %s\nBalance: %s", metadata.TokenSymbol, metadata.Balance) | ||||
| 
 | ||||
| 	return res, nil | ||||
| } | ||||
| @ -1720,10 +1763,9 @@ func (h *Handlers) GetVoucherDetails(ctx context.Context, sym string, input []by | ||||
| 		return res, nil | ||||
| 	} | ||||
| 
 | ||||
| 	tokenSymbol := voucherData.TokenSymbol | ||||
| 	tokenName := voucherData.TokenName | ||||
| 
 | ||||
| 	res.Content = fmt.Sprintf("%s %s", tokenSymbol, tokenName) | ||||
| 	res.Content = fmt.Sprintf( | ||||
| 		"Name: %s\nSymbol: %s\nCommodity: %s\nLocation: %s", voucherData.TokenName, voucherData.TokenSymbol, voucherData.TokenCommodity, voucherData.TokenLocation, | ||||
| 	) | ||||
| 
 | ||||
| 	return res, nil | ||||
| } | ||||
|  | ||||
| @ -1498,10 +1498,10 @@ func TestValidateRecipient(t *testing.T) { | ||||
| 	}{ | ||||
| 		{ | ||||
| 			name:  "Test with invalid recepient", | ||||
| 			input: []byte("9234adf5"), | ||||
| 			input: []byte("7?1234"), | ||||
| 			expectedResult: resource.Result{ | ||||
| 				FlagSet: []uint32{flag_invalid_recipient}, | ||||
| 				Content: "9234adf5", | ||||
| 				Content: "7?1234", | ||||
| 			}, | ||||
| 		}, | ||||
| 		{ | ||||
| @ -1517,22 +1517,40 @@ func TestValidateRecipient(t *testing.T) { | ||||
| 			input:          []byte("0711223344"), | ||||
| 			expectedResult: resource.Result{}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:           "Test with address", | ||||
| 			input:          []byte("0xd4c288865Ce0985a481Eef3be02443dF5E2e4Ea9"), | ||||
| 			expectedResult: resource.Result{}, | ||||
| 		}, | ||||
| 		{ | ||||
| 			name:           "Test with alias recepient", | ||||
| 			input:          []byte("alias123"), | ||||
| 			expectedResult: resource.Result{}, | ||||
| 		}, | ||||
| 	} | ||||
| 
 | ||||
| 	// store a public key for the valid recipient
 | ||||
| 	err = store.WriteEntry(ctx, "0711223344", common.DATA_PUBLIC_KEY, []byte(publicKey)) | ||||
| 	err = store.WriteEntry(ctx, "+254711223344", common.DATA_PUBLIC_KEY, []byte(publicKey)) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 
 | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			mockAccountService := new(mocks.MockAccountService) | ||||
| 			// Create the Handlers instance
 | ||||
| 			h := &Handlers{ | ||||
| 				flagManager:   fm.parser, | ||||
| 				userdataStore: store, | ||||
| 				flagManager:    fm.parser, | ||||
| 				userdataStore:  store, | ||||
| 				accountService: mockAccountService, | ||||
| 			} | ||||
| 
 | ||||
| 			aliasResponse := &dataserviceapi.AliasAddress{ | ||||
| 				Address: "0xd4c288865Ce0985a481Eef3be02443dF5E2e4Ea9", | ||||
| 			} | ||||
| 
 | ||||
| 			mockAccountService.On("CheckAliasAddress", string(tt.input)).Return(aliasResponse, nil) | ||||
| 
 | ||||
| 			// Call the method
 | ||||
| 			res, err := h.ValidateRecipient(ctx, "validate_recepient", tt.input) | ||||
| 
 | ||||
| @ -1564,7 +1582,7 @@ func TestCheckBalance(t *testing.T) { | ||||
| 			publicKey:      "0X98765432109", | ||||
| 			activeSym:      "ETH", | ||||
| 			activeBal:      "1.5", | ||||
| 			expectedResult: resource.Result{Content: "Balance: 1.5 ETH\n"}, | ||||
| 			expectedResult: resource.Result{Content: "Balance: 1.50 ETH\n"}, | ||||
| 			expectError:    false, | ||||
| 		}, | ||||
| 	} | ||||
| @ -1990,7 +2008,7 @@ func TestViewVoucher(t *testing.T) { | ||||
| 
 | ||||
| 	res, err := h.ViewVoucher(ctx, "view_voucher", []byte("1")) | ||||
| 	assert.NoError(t, err) | ||||
| 	assert.Equal(t, res.Content, "SRF\n100") | ||||
| 	assert.Equal(t, res.Content, "Symbol: SRF\nBalance: 100") | ||||
| } | ||||
| 
 | ||||
| func TestSetVoucher(t *testing.T) { | ||||
| @ -2024,3 +2042,42 @@ func TestSetVoucher(t *testing.T) { | ||||
| 
 | ||||
| 	assert.Equal(t, string(tempData.TokenSymbol), res.Content) | ||||
| } | ||||
| 
 | ||||
| func TestGetVoucherDetails(t *testing.T) { | ||||
| 	ctx, store := InitializeTestStore(t) | ||||
| 	fm, err := NewFlagManager(flagsPath) | ||||
| 	if err != nil { | ||||
| 		t.Logf(err.Error()) | ||||
| 	} | ||||
| 	mockAccountService := new(mocks.MockAccountService) | ||||
| 
 | ||||
| 	sessionId := "session123" | ||||
| 	ctx = context.WithValue(ctx, "SessionId", sessionId) | ||||
| 	expectedResult := resource.Result{} | ||||
| 
 | ||||
| 	tokA_AAddress := "0x0000000000000000000000000000000000000000" | ||||
| 
 | ||||
| 	h := &Handlers{ | ||||
| 		userdataStore:  store, | ||||
| 		flagManager:    fm.parser, | ||||
| 		accountService: mockAccountService, | ||||
| 	} | ||||
| 	err = store.WriteEntry(ctx, sessionId, common.DATA_ACTIVE_ADDRESS, []byte(tokA_AAddress)) | ||||
| 	if err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	tokenDetails := &models.VoucherDataResult{ | ||||
| 		TokenName:      "Token A", | ||||
| 		TokenSymbol:    "TOKA", | ||||
| 		TokenLocation:  "Kilifi,Kenya", | ||||
| 		TokenCommodity: "Farming", | ||||
| 	} | ||||
| 	expectedResult.Content = fmt.Sprintf( | ||||
| 		"Name: %s\nSymbol: %s\nCommodity: %s\nLocation: %s", tokenDetails.TokenName, tokenDetails.TokenSymbol, tokenDetails.TokenCommodity, tokenDetails.TokenLocation, | ||||
| 	) | ||||
| 	mockAccountService.On("VoucherData", string(tokA_AAddress)).Return(tokenDetails, nil) | ||||
| 
 | ||||
| 	res, err := h.GetVoucherDetails(ctx, "SessionId", []byte("")) | ||||
| 	assert.NoError(t, err) | ||||
| 	assert.Equal(t, expectedResult, res) | ||||
| } | ||||
|  | ||||
| @ -49,6 +49,6 @@ func (m *MockAccountService) TokenTransfer(ctx context.Context, amount, from, to | ||||
| } | ||||
| 
 | ||||
| func (m *MockAccountService) CheckAliasAddress(ctx context.Context, alias string) (*dataserviceapi.AliasAddress, error) { | ||||
| 	args := m.Called() | ||||
| 	args := m.Called(alias) | ||||
| 	return args.Get(0).(*dataserviceapi.AliasAddress), args.Error(1) | ||||
| } | ||||
|  | ||||
| @ -1,6 +1,9 @@ | ||||
| package utils | ||||
| 
 | ||||
| import "time" | ||||
| import ( | ||||
| 	"strconv" | ||||
| 	"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.
 | ||||
| @ -25,11 +28,29 @@ func CalculateAge(birthdate, today time.Time) int { | ||||
| // It subtracts the YOB from the current year to determine the age.
 | ||||
| //
 | ||||
| // Parameters:
 | ||||
| //   yob: The year of birth as an integer.
 | ||||
| //
 | ||||
| //	yob: The year of birth as an integer.
 | ||||
| //
 | ||||
| // Returns:
 | ||||
| //   The calculated age as an integer.
 | ||||
| //
 | ||||
| //	The calculated age as an integer.
 | ||||
| func CalculateAgeWithYOB(yob int) int { | ||||
|     currentYear := time.Now().Year() | ||||
|     return currentYear - yob | ||||
| } | ||||
| 	currentYear := time.Now().Year() | ||||
| 	return currentYear - yob | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| //IsValidYob checks if the provided yob can be considered valid 
 | ||||
| func IsValidYOb(yob string) bool { | ||||
| 	currentYear := time.Now().Year() | ||||
| 	yearOfBirth, err := strconv.ParseInt(yob, 10, 64) | ||||
| 	if err != nil { | ||||
| 		return false | ||||
| 	} | ||||
| 	if yearOfBirth >= 1900 && int(yearOfBirth) <= currentYear { | ||||
| 		return true | ||||
| 	} else { | ||||
| 		return false | ||||
| 	} | ||||
| 
 | ||||
| } | ||||
|  | ||||
							
								
								
									
										17
									
								
								internal/utils/name.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								internal/utils/name.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,17 @@ | ||||
| package utils | ||||
| 
 | ||||
| func ConstructName(firstName, familyName, defaultValue string) string { | ||||
| 	name := defaultValue | ||||
| 	if familyName != defaultValue { | ||||
| 		if firstName != defaultValue { | ||||
| 			name = firstName + " " + familyName | ||||
| 		} else { | ||||
| 			name = familyName | ||||
| 		} | ||||
| 	} else { | ||||
| 		if firstName != defaultValue { | ||||
| 			name = firstName | ||||
| 		} | ||||
| 	} | ||||
| 	return name | ||||
| } | ||||
| @ -54,7 +54,7 @@ | ||||
|                 }, | ||||
|                 { | ||||
|                     "input": "1235", | ||||
|                     "expectedContent": "Incorrect pin\n1:Retry\n9:Quit" | ||||
|                     "expectedContent": "Incorrect PIN\n1:Retry\n9:Quit" | ||||
|                 }, | ||||
|                 { | ||||
|                     "input": "1", | ||||
| @ -62,7 +62,7 @@ | ||||
|                 }, | ||||
|                 { | ||||
|                     "input": "1234", | ||||
|                     "expectedContent": "Select language:\n0:english\n1:kiswahili" | ||||
|                     "expectedContent": "Select language:\n0:English\n1:Kiswahili" | ||||
|                 }, | ||||
|                 { | ||||
|                     "input": "0", | ||||
| @ -95,7 +95,7 @@ | ||||
|                 }, | ||||
|                 { | ||||
|                     "input": "1235", | ||||
|                     "expectedContent": "Incorrect pin\n1:Retry\n9:Quit" | ||||
|                     "expectedContent": "Incorrect PIN\n1:Retry\n9:Quit" | ||||
|                 }, | ||||
|                 { | ||||
|                     "input": "1", | ||||
| @ -141,7 +141,7 @@ | ||||
|                 }, | ||||
|                 { | ||||
|                     "input": "1235", | ||||
|                     "expectedContent": "Incorrect pin\n1:Retry\n9:Quit" | ||||
|                     "expectedContent": "Incorrect PIN\n1:Retry\n9:Quit" | ||||
|                 }, | ||||
|                 { | ||||
|                     "input": "1", | ||||
|  | ||||
| @ -7,11 +7,11 @@ | ||||
|                 "steps": [ | ||||
|                     { | ||||
|                         "input": "", | ||||
|                         "expectedContent": "Welcome to Sarafu Network\nPlease select a language\n0:english\n1:kiswahili" | ||||
|                         "expectedContent": "Welcome to Sarafu Network\nPlease select a language\n0:English\n1:Kiswahili" | ||||
|                     }, | ||||
|                     { | ||||
|                         "input": "0", | ||||
|                         "expectedContent": "Do you agree to terms and conditions?\n0:yes\n1:no" | ||||
|                         "expectedContent": "Do you agree to terms and conditions?\n0:Yes\n1:No" | ||||
|                     }, | ||||
|                     { | ||||
|                         "input": "0", | ||||
| @ -40,11 +40,11 @@ | ||||
|                 "steps": [ | ||||
|                     { | ||||
|                         "input": "", | ||||
|                         "expectedContent": "Welcome to Sarafu Network\nPlease select a language\n0:english\n1:kiswahili" | ||||
|                         "expectedContent": "Welcome to Sarafu Network\nPlease select a language\n0:English\n1:Kiswahili" | ||||
|                     }, | ||||
|                     { | ||||
|                         "input": "0", | ||||
|                         "expectedContent": "Do you agree to terms and conditions?\n0:yes\n1:no" | ||||
|                         "expectedContent": "Do you agree to terms and conditions?\n0:Yes\n1:No" | ||||
|                     }, | ||||
|                     { | ||||
|                         "input": "1", | ||||
|  | ||||
| @ -1,8 +1,10 @@ | ||||
| package models | ||||
| 
 | ||||
| type VoucherDataResult struct { | ||||
| 	TokenName     string `json:"tokenName"` | ||||
| 	TokenSymbol   string `json:"tokenSymbol"` | ||||
| 	TokenDecimals int    `json:"tokenDecimals"` | ||||
| 	SinkAddress   string `json:"sinkAddress"` | ||||
| 	TokenName      string `json:"tokenName"` | ||||
| 	TokenSymbol    string `json:"tokenSymbol"` | ||||
| 	TokenDecimals  int    `json:"tokenDecimals"` | ||||
| 	SinkAddress    string `json:"sinkAddress"` | ||||
| 	TokenCommodity string `json:"tokenCommodity"` | ||||
| 	TokenLocation  string `json:"tokenLocation"` | ||||
| } | ||||
|  | ||||
							
								
								
									
										1
									
								
								services/registration/_catch_swa
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								services/registration/_catch_swa
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | ||||
| Tatizo la kimtambo limetokea,tafadhali jaribu tena baadaye. | ||||
							
								
								
									
										1
									
								
								services/registration/english_menu
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								services/registration/english_menu
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | ||||
| English | ||||
| @ -1 +1 @@ | ||||
| Incorrect pin | ||||
| Incorrect PIN | ||||
| @ -1 +1 @@ | ||||
| PIN mpya na udhibitisho wa pin mpya hazilingani.Tafadhali jaribu tena.Kwa usaidizi piga simu +254757628885. | ||||
| PIN mpya na udhibitisho wa PIN mpya hazilingani.Tafadhali jaribu tena.Kwa usaidizi piga simu +254757628885. | ||||
|  | ||||
							
								
								
									
										1
									
								
								services/registration/kiswahili_menu
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								services/registration/kiswahili_menu
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | ||||
| Kiswahili | ||||
| @ -13,7 +13,7 @@ msgstr "Kwa usaidizi zaidi,piga: 0757628885" | ||||
| msgid "Balance: %s\n" | ||||
| msgstr "Salio: %s\n" | ||||
| 
 | ||||
| msid "Your invite request for %s to Sarafu Network failed. Please try again later." | ||||
| msgid "Your invite request for %s to Sarafu Network failed. Please try again later." | ||||
| msgstr "Ombi lako la kumwalika %s kwa matandao wa Sarafu halikufaulu. Tafadhali jaribu tena baadaye." | ||||
| 
 | ||||
| msgid "Your invitation to %s to join Sarafu Network has been sent." | ||||
| @ -23,4 +23,7 @@ msgid "Your request failed. Please try again later." | ||||
| msgstr "Ombi lako halikufaulu. Tafadhali jaribu tena baadaye." | ||||
| 
 | ||||
| msgid "Community Balance: 0.00" | ||||
| msgid "Salio la Kikundi: 0.00" | ||||
| msgstr "Salio la Kikundi: 0.00" | ||||
| 
 | ||||
| msgid "Symbol: %s\nBalance: %s" | ||||
| msgstr "Sarafu: %s\nSalio: %s" | ||||
|  | ||||
| @ -1,10 +1,10 @@ | ||||
| LOAD set_default_voucher 8 | ||||
| RELOAD set_default_voucher | ||||
| LOAD check_balance 64 | ||||
| RELOAD check_balance | ||||
| LOAD check_vouchers 10 | ||||
| RELOAD check_vouchers | ||||
| CATCH api_failure  flag_api_call_error  1 | ||||
| LOAD check_balance 128 | ||||
| RELOAD check_balance | ||||
| CATCH api_failure flag_api_call_error 1 | ||||
| MAP check_balance | ||||
| MOUT send 1 | ||||
| MOUT vouchers 2 | ||||
|  | ||||
| @ -1 +1 @@ | ||||
| Salio lako ni: 0.00 SRF | ||||
| {{.check_balance}} | ||||
							
								
								
									
										1
									
								
								services/registration/next_menu
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								services/registration/next_menu
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | ||||
| Next | ||||
							
								
								
									
										1
									
								
								services/registration/next_menu_swa
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								services/registration/next_menu_swa
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | ||||
| Mbele | ||||
| @ -1 +1 @@ | ||||
| no | ||||
| No | ||||
| @ -1 +1 @@ | ||||
| la | ||||
| La | ||||
							
								
								
									
										1
									
								
								services/registration/prev_menu
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								services/registration/prev_menu
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | ||||
| Prev | ||||
							
								
								
									
										1
									
								
								services/registration/prev_menu_swa
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								services/registration/prev_menu_swa
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | ||||
| Nyuma | ||||
| @ -1,3 +1,4 @@ | ||||
| LOAD set_language 6 | ||||
| RELOAD set_language | ||||
| CATCH terms flag_account_created 0 | ||||
| MOVE language_changed | ||||
|  | ||||
| @ -1,3 +1,4 @@ | ||||
| LOAD set_language 6 | ||||
| RELOAD set_language | ||||
| CATCH terms flag_account_created 0 | ||||
| MOVE language_changed | ||||
|  | ||||
| @ -1 +1 @@ | ||||
| yes | ||||
| Yes | ||||
| @ -1 +1 @@ | ||||
| ndio | ||||
| Ndio | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user