489 lines
17 KiB
Go
489 lines
17 KiB
Go
// Copyright 2017-2021 DERO Project. All rights reserved.
|
|
// Use of this source code in any form is governed by RESEARCH license.
|
|
// license can be found in the LICENSE file.
|
|
// GPG: 0F39 E425 8C65 3947 702A 8234 08B2 0360 A03A 9DE8
|
|
//
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
|
|
// EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
|
// THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
// STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
|
// THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
package main
|
|
|
|
import "io"
|
|
import "os"
|
|
import "time"
|
|
import "fmt"
|
|
import "errors"
|
|
|
|
import "strings"
|
|
|
|
import "path/filepath"
|
|
import "encoding/json"
|
|
|
|
import "github.com/chzyer/readline"
|
|
|
|
import "github.com/deroproject/derohe/rpc"
|
|
import "github.com/deroproject/derohe/globals"
|
|
|
|
//import "github.com/deroproject/derohe/address"
|
|
|
|
import "github.com/deroproject/derohe/cryptography/crypto"
|
|
import "github.com/deroproject/derohe/transaction"
|
|
|
|
// handle menu if a wallet is currently opened
|
|
func display_easymenu_post_open_command(l *readline.Instance) {
|
|
w := l.Stderr()
|
|
io.WriteString(w, "Menu:\n")
|
|
|
|
io.WriteString(w, "\t\033[1m1\033[0m\tDisplay account Address \n")
|
|
io.WriteString(w, "\t\033[1m2\033[0m\tDisplay Seed "+color_red+"(Please save seed in safe location)\n\033[0m")
|
|
|
|
io.WriteString(w, "\t\033[1m3\033[0m\tDisplay Keys (hex)\n")
|
|
|
|
if !wallet.IsRegistered() {
|
|
io.WriteString(w, "\t\033[1m4\033[0m\tAccount registration to blockchain (registration has no fee requirement and is precondition to use the account)\n")
|
|
io.WriteString(w, "\n")
|
|
io.WriteString(w, "\n")
|
|
} else { // hide some commands, if view only wallet
|
|
io.WriteString(w, "\t\033[1m4\033[0m\tDisplay wallet pool\n")
|
|
io.WriteString(w, "\t\033[1m5\033[0m\tTransfer (send DERO) to Another Wallet\n")
|
|
io.WriteString(w, "\t\033[1m6\033[0m\tToken transfer to another wallet\n")
|
|
io.WriteString(w, "\n")
|
|
}
|
|
|
|
io.WriteString(w, "\t\033[1m7\033[0m\tChange wallet password\n")
|
|
io.WriteString(w, "\t\033[1m8\033[0m\tClose Wallet\n")
|
|
if wallet.IsRegistered() {
|
|
io.WriteString(w, "\t\033[1m12\033[0m\tTransfer all balance (send DERO) To Another Wallet\n")
|
|
io.WriteString(w, "\t\033[1m13\033[0m\tShow transaction history\n")
|
|
io.WriteString(w, "\t\033[1m14\033[0m\tRescan transaction history\n")
|
|
io.WriteString(w, "\t\033[1m15\033[0m\tExport all transaction history in json format\n")
|
|
}
|
|
|
|
io.WriteString(w, "\n\t\033[1m9\033[0m\tExit menu and start prompt\n")
|
|
io.WriteString(w, "\t\033[1m0\033[0m\tExit Wallet\n")
|
|
|
|
}
|
|
|
|
// this handles all the commands if wallet in menu mode and a wallet is opened
|
|
func handle_easymenu_post_open_command(l *readline.Instance, line string) (processed bool) {
|
|
|
|
var err error
|
|
_ = err
|
|
line = strings.TrimSpace(line)
|
|
line_parts := strings.Fields(line)
|
|
processed = true
|
|
|
|
if len(line_parts) < 1 { // if no command return
|
|
return
|
|
}
|
|
|
|
command := ""
|
|
if len(line_parts) >= 1 {
|
|
command = strings.ToLower(line_parts[0])
|
|
}
|
|
|
|
offline_tx := false
|
|
_ = offline_tx
|
|
switch command {
|
|
case "1":
|
|
fmt.Fprintf(l.Stderr(), "Wallet address : "+color_green+"%s"+color_white+"\n", wallet.GetAddress())
|
|
|
|
if !wallet.IsRegistered() {
|
|
reg_tx := wallet.GetRegistrationTX()
|
|
fmt.Fprintf(l.Stderr(), "Registration TX : "+color_green+"%x"+color_white+"\n", reg_tx.Serialize())
|
|
}
|
|
PressAnyKey(l, wallet)
|
|
|
|
case "2": // give user his seed
|
|
if !ValidateCurrentPassword(l, wallet) {
|
|
logger.Error(fmt.Errorf("Invalid password"), "")
|
|
PressAnyKey(l, wallet)
|
|
break
|
|
}
|
|
display_seed(l, wallet) // seed should be given only to authenticated users
|
|
PressAnyKey(l, wallet)
|
|
|
|
case "3": // give user his keys in hex form
|
|
|
|
if !ValidateCurrentPassword(l, wallet) {
|
|
logger.Error(fmt.Errorf("Invalid password"), "")
|
|
PressAnyKey(l, wallet)
|
|
break
|
|
}
|
|
|
|
display_spend_key(l, wallet)
|
|
PressAnyKey(l, wallet)
|
|
|
|
case "4": // Registration
|
|
|
|
if !wallet.IsRegistered() {
|
|
|
|
fmt.Fprintf(l.Stderr(), "Wallet address : "+color_green+"%s"+color_white+" is going to be registered.This is a pre-condition for using the online chain.It will take few seconds to register.\n", wallet.GetAddress())
|
|
|
|
// at this point we must send the registration transaction
|
|
fmt.Fprintf(l.Stderr(), "Wallet address : "+color_green+"%s"+color_white+" is going to be registered.Pls wait till the account is registered.\n", wallet.GetAddress())
|
|
|
|
fmt.Fprintf(l.Stderr(), "This will take a couple of minutes.Please wait....\n")
|
|
|
|
var reg_tx *transaction.Transaction
|
|
for {
|
|
|
|
reg_tx = wallet.GetRegistrationTX()
|
|
hash := reg_tx.GetHash()
|
|
|
|
if hash[0] == 0 && hash[1] == 0 && hash[2] <= 0x3 {
|
|
break
|
|
}
|
|
}
|
|
fmt.Fprintf(l.Stderr(), "Registration TXID %s\n", reg_tx.GetHash())
|
|
err := wallet.SendTransaction(reg_tx)
|
|
if err != nil {
|
|
fmt.Fprintf(l.Stderr(), "sending registration tx err %s\n", err)
|
|
} else {
|
|
fmt.Fprintf(l.Stderr(), "registration tx dispatched successfully\n")
|
|
}
|
|
|
|
} else {
|
|
|
|
}
|
|
|
|
case "6":
|
|
if !valid_registration_or_display_error(l, wallet) {
|
|
break
|
|
}
|
|
if !ValidateCurrentPassword(l, wallet) {
|
|
logger.Error(fmt.Errorf("Invalid password"), "")
|
|
break
|
|
}
|
|
|
|
scid, err := ReadSCID(l)
|
|
if err != nil {
|
|
logger.Error(err, "error reading SCID")
|
|
break
|
|
}
|
|
|
|
a, err := ReadAddress(l, wallet)
|
|
if err != nil {
|
|
logger.Error(err, "error reading address")
|
|
break
|
|
}
|
|
|
|
var amount_to_transfer uint64
|
|
|
|
amount_str := read_line_with_prompt(l, fmt.Sprintf("Enter token amount to transfer in SCID (max TODO): "))
|
|
if amount_str == "" {
|
|
amount_str = ".00001"
|
|
}
|
|
amount_to_transfer, err = globals.ParseAmount(amount_str)
|
|
if err != nil {
|
|
logger.Error(err, "Err parsing amount")
|
|
break // invalid amount provided, bail out
|
|
}
|
|
|
|
if ConfirmYesNoDefaultNo(l, "Confirm Transaction (y/N)") {
|
|
tx, err := wallet.TransferPayload0([]rpc.Transfer{rpc.Transfer{SCID: scid, Amount: amount_to_transfer, Destination: a.String()}}, 0, false, rpc.Arguments{}, 0, false) // empty SCDATA
|
|
|
|
if err != nil {
|
|
logger.Error(err, "Error while building Transaction")
|
|
break
|
|
}
|
|
if err = wallet.SendTransaction(tx); err != nil {
|
|
logger.Error(err, "Error while dispatching Transaction")
|
|
break
|
|
}
|
|
logger.Info("Dispatched tx", "txid", tx.GetHash().String())
|
|
}
|
|
|
|
case "5":
|
|
if !valid_registration_or_display_error(l, wallet) {
|
|
break
|
|
}
|
|
if !ValidateCurrentPassword(l, wallet) {
|
|
logger.Error(fmt.Errorf("Invalid password"), "")
|
|
break
|
|
}
|
|
|
|
// a , amount_to_transfer, err := collect_transfer_info(l,wallet)
|
|
a, err := ReadAddress(l, wallet)
|
|
if err != nil {
|
|
logger.Error(err, "error reading address")
|
|
break
|
|
}
|
|
|
|
var amount_to_transfer uint64
|
|
|
|
var arguments = rpc.Arguments{
|
|
// { rpc.RPC_DESTINATION_PORT, rpc.DataUint64,uint64(0x1234567812345678)},
|
|
// { rpc.RPC_VALUE_TRANSFER, rpc.DataUint64,uint64(12345)},
|
|
// { rpc.RPC_EXPIRY , rpc.DataTime, time.Now().Add(time.Hour).UTC()},
|
|
// { rpc.RPC_COMMENT , rpc.DataString, "Purchase XYZ"},
|
|
}
|
|
if a.IsIntegratedAddress() { // read everything from the address
|
|
|
|
if a.Arguments.Validate_Arguments() != nil {
|
|
logger.Error(err, "Integrated Address arguments could not be validated.")
|
|
break
|
|
}
|
|
|
|
if !a.Arguments.Has(rpc.RPC_DESTINATION_PORT, rpc.DataUint64) { // but only it is present
|
|
logger.Error(fmt.Errorf("Integrated Address does not contain destination port."), "")
|
|
break
|
|
}
|
|
|
|
arguments = append(arguments, rpc.Argument{Name: rpc.RPC_DESTINATION_PORT, DataType: rpc.DataUint64, Value: a.Arguments.Value(rpc.RPC_DESTINATION_PORT, rpc.DataUint64).(uint64)})
|
|
// arguments = append(arguments, rpc.Argument{"Comment", rpc.DataString, "holygrail of all data is now working if you can see this"})
|
|
|
|
if a.Arguments.Has(rpc.RPC_EXPIRY, rpc.DataTime) { // but only it is present
|
|
|
|
if a.Arguments.Value(rpc.RPC_EXPIRY, rpc.DataTime).(time.Time).Before(time.Now().UTC()) {
|
|
logger.Error(nil, "This address has expired.", "expiry time", a.Arguments.Value(rpc.RPC_EXPIRY, rpc.DataTime))
|
|
break
|
|
} else {
|
|
logger.Info("This address will expire ", "expiry time", a.Arguments.Value(rpc.RPC_EXPIRY, rpc.DataTime))
|
|
}
|
|
}
|
|
|
|
logger.Info("Destination port is integreted in address.", "dst port", a.Arguments.Value(rpc.RPC_DESTINATION_PORT, rpc.DataUint64).(uint64))
|
|
|
|
if a.Arguments.Has(rpc.RPC_COMMENT, rpc.DataString) { // but only it is present
|
|
logger.Info("Integrated Message", "comment", a.Arguments.Value(rpc.RPC_COMMENT, rpc.DataString))
|
|
arguments = append(arguments, rpc.Argument{rpc.RPC_COMMENT, rpc.DataString, a.Arguments.Value(rpc.RPC_COMMENT, rpc.DataString)})
|
|
}
|
|
}
|
|
|
|
// arguments have been already validated
|
|
for _, arg := range a.Arguments {
|
|
if !(arg.Name == rpc.RPC_COMMENT || arg.Name == rpc.RPC_EXPIRY || arg.Name == rpc.RPC_DESTINATION_PORT || arg.Name == rpc.RPC_SOURCE_PORT || arg.Name == rpc.RPC_VALUE_TRANSFER || arg.Name == rpc.RPC_NEEDS_REPLYBACK_ADDRESS) {
|
|
switch arg.DataType {
|
|
case rpc.DataString:
|
|
if v, err := ReadString(l, arg.Name, arg.Value.(string)); err == nil {
|
|
arguments = append(arguments, rpc.Argument{Name: arg.Name, DataType: arg.DataType, Value: v})
|
|
} else {
|
|
logger.Error(fmt.Errorf("%s could not be parsed (type %s),", arg.Name, arg.DataType), "")
|
|
return
|
|
}
|
|
case rpc.DataInt64:
|
|
if v, err := ReadInt64(l, arg.Name, arg.Value.(int64)); err == nil {
|
|
arguments = append(arguments, rpc.Argument{Name: arg.Name, DataType: arg.DataType, Value: v})
|
|
} else {
|
|
logger.Error(fmt.Errorf("%s could not be parsed (type %s),", arg.Name, arg.DataType), "")
|
|
return
|
|
}
|
|
case rpc.DataUint64:
|
|
if v, err := ReadUint64(l, arg.Name, arg.Value.(uint64)); err == nil {
|
|
arguments = append(arguments, rpc.Argument{Name: arg.Name, DataType: arg.DataType, Value: v})
|
|
} else {
|
|
logger.Error(fmt.Errorf("%s could not be parsed (type %s),", arg.Name, arg.DataType), "")
|
|
return
|
|
}
|
|
case rpc.DataFloat64:
|
|
if v, err := ReadFloat64(l, arg.Name, arg.Value.(float64)); err == nil {
|
|
arguments = append(arguments, rpc.Argument{Name: arg.Name, DataType: arg.DataType, Value: v})
|
|
} else {
|
|
logger.Error(fmt.Errorf("%s could not be parsed (type %s),", arg.Name, arg.DataType), "")
|
|
return
|
|
}
|
|
case rpc.DataTime:
|
|
logger.Error(fmt.Errorf("time argument is currently not supported."), "")
|
|
break
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
if a.Arguments.Has(rpc.RPC_VALUE_TRANSFER, rpc.DataUint64) { // but only it is present
|
|
logger.Info("Transaction", "Value", globals.FormatMoney(a.Arguments.Value(rpc.RPC_VALUE_TRANSFER, rpc.DataUint64).(uint64)))
|
|
amount_to_transfer = a.Arguments.Value(rpc.RPC_VALUE_TRANSFER, rpc.DataUint64).(uint64)
|
|
} else {
|
|
|
|
mbal, _ := wallet.Get_Balance()
|
|
amount_str := read_line_with_prompt(l, fmt.Sprintf("Enter amount to transfer in DERO (current balance %s): ", globals.FormatMoney(mbal)))
|
|
|
|
if amount_str == "" {
|
|
logger.Error(nil, "Cannot transfer 0")
|
|
break // invalid amount provided, bail out
|
|
}
|
|
amount_to_transfer, err = globals.ParseAmount(amount_str)
|
|
if err != nil {
|
|
logger.Error(err, "Err parsing amount")
|
|
break // invalid amount provided, bail out
|
|
}
|
|
}
|
|
|
|
// check whether the service needs the address of sender
|
|
// this is required to enable services which are completely invisisble to external entities
|
|
// external entities means anyone except sender/receiver
|
|
if a.Arguments.Has(rpc.RPC_NEEDS_REPLYBACK_ADDRESS, rpc.DataUint64) {
|
|
logger.Info("This RPC has requested your address.")
|
|
logger.Info("If you are expecting something back, it needs to be sent")
|
|
logger.Info("Your address will remain completely invisible to external entities(only sender/receiver can see your address)")
|
|
arguments = append(arguments, rpc.Argument{Name: rpc.RPC_REPLYBACK_ADDRESS, DataType: rpc.DataAddress, Value: wallet.GetAddress()})
|
|
}
|
|
|
|
// if no arguments, use space by embedding a small comment
|
|
if len(arguments) == 0 { // allow user to enter Comment
|
|
if v, err := ReadUint64(l, "Please enter payment id (or destination port number)", uint64(0)); err == nil {
|
|
arguments = append(arguments, rpc.Argument{Name: rpc.RPC_DESTINATION_PORT, DataType: rpc.DataUint64, Value: v})
|
|
} else {
|
|
logger.Error(err, fmt.Sprintf("%s could not be parsed (type %s),", "Number", rpc.DataUint64))
|
|
return
|
|
}
|
|
|
|
if v, err := ReadString(l, "Comment", ""); err == nil {
|
|
arguments = append(arguments, rpc.Argument{Name: rpc.RPC_COMMENT, DataType: rpc.DataString, Value: v})
|
|
} else {
|
|
logger.Error(fmt.Errorf("%s could not be parsed (type %s),", "Comment", rpc.DataString), "")
|
|
return
|
|
}
|
|
}
|
|
|
|
if _, err := arguments.CheckPack(transaction.PAYLOAD0_LIMIT); err != nil {
|
|
logger.Error(err, "Arguments packing err")
|
|
return
|
|
}
|
|
|
|
if ConfirmYesNoDefaultNo(l, "Confirm Transaction (y/N)") {
|
|
|
|
//src_port := uint64(0xffffffffffffffff)
|
|
|
|
tx, err := wallet.TransferPayload0([]rpc.Transfer{rpc.Transfer{Amount: amount_to_transfer, Destination: a.String(), Payload_RPC: arguments}}, 0, false, rpc.Arguments{}, 0, false) // empty SCDATA
|
|
|
|
if err != nil {
|
|
logger.Error(err, "Error while building Transaction")
|
|
break
|
|
}
|
|
|
|
if err = wallet.SendTransaction(tx); err != nil {
|
|
logger.Error(err, "Error while dispatching Transaction")
|
|
break
|
|
}
|
|
logger.Info("Dispatched tx", "txid", tx.GetHash().String())
|
|
//fmt.Printf("queued tx err %s\n")
|
|
}
|
|
|
|
case "12":
|
|
if !valid_registration_or_display_error(l, wallet) {
|
|
break
|
|
}
|
|
if !ValidateCurrentPassword(l, wallet) {
|
|
logger.Error(fmt.Errorf("Invalid password"), "")
|
|
break
|
|
}
|
|
|
|
logger.Error(err, "Not supported ")
|
|
|
|
/*
|
|
// a , amount_to_transfer, err := collect_transfer_info(l,wallet)
|
|
fmt.Printf("dest address %s\n", "deroi1qxqqkmaz8nhv4q07w3cjyt84kmrqnuw4nprpqfl9xmmvtvwa7cdykxq5dph4ufnx5ndq4ltraf (14686f5e2666a4da) dero1qxqqkmaz8nhv4q07w3cjyt84kmrqnuw4nprpqfl9xmmvtvwa7cdykxqpfpaes")
|
|
a, err := ReadAddress(l)
|
|
if err != nil {
|
|
globals.Logger.Warnf("Err :%s", err)
|
|
break
|
|
}
|
|
// if user provided an integrated address donot ask him payment id
|
|
if a.IsIntegratedAddress() {
|
|
globals.Logger.Infof("Payment ID is integreted in address ID:%x", a.PaymentID)
|
|
}
|
|
|
|
if ConfirmYesNoDefaultNo(l, "Confirm Transaction to send entire balance (y/N)") {
|
|
|
|
addr_list := []address.Address{*a}
|
|
amount_list := []uint64{0} // transfer 50 dero, 2 dero
|
|
fees_per_kb := uint64(0) // fees must be calculated by walletapi
|
|
uid, err := wallet.PoolTransfer(addr_list, amount_list, fees_per_kb, 0, true)
|
|
_ = uid
|
|
if err != nil {
|
|
globals.Logger.Warnf("Error while building Transaction err %s\n", err)
|
|
break
|
|
}
|
|
}
|
|
*/
|
|
|
|
//PressAnyKey(l, wallet) // wait for a key press
|
|
|
|
case "7": // change password
|
|
if ConfirmYesNoDefaultNo(l, "Change wallet password (y/N)") &&
|
|
ValidateCurrentPassword(l, wallet) {
|
|
|
|
new_password := ReadConfirmedPassword(l, "Enter new password", "Confirm password")
|
|
err = wallet.Set_Encrypted_Wallet_Password(new_password)
|
|
if err == nil {
|
|
logger.Info("Wallet password successfully changed")
|
|
} else {
|
|
logger.Error(err, "Wallet password could not be changed")
|
|
}
|
|
}
|
|
|
|
case "8": // close and discard user key
|
|
|
|
wallet.Close_Encrypted_Wallet()
|
|
prompt_mutex.Lock()
|
|
wallet = nil // overwrite previous instance
|
|
prompt_mutex.Unlock()
|
|
|
|
fmt.Fprintf(l.Stderr(), color_yellow+"Wallet closed"+color_white)
|
|
|
|
case "9": // enable prompt mode
|
|
menu_mode = false
|
|
logger.Info("Prompt mode enabled, type \"menu\" command to start menu mode")
|
|
|
|
case "0", "bye", "exit", "quit":
|
|
wallet.Close_Encrypted_Wallet() // save the wallet
|
|
prompt_mutex.Lock()
|
|
wallet = nil
|
|
globals.Exit_In_Progress = true
|
|
prompt_mutex.Unlock()
|
|
fmt.Fprintf(l.Stderr(), color_yellow+"Wallet closed"+color_white)
|
|
fmt.Fprintf(l.Stderr(), color_yellow+"Exiting"+color_white)
|
|
|
|
case "13":
|
|
var zeroscid crypto.Hash
|
|
show_transfers(l, wallet, zeroscid, 100)
|
|
|
|
case "14":
|
|
logger.Info("Rescanning wallet history")
|
|
rescan_bc(wallet)
|
|
case "15":
|
|
if !ValidateCurrentPassword(l, wallet) {
|
|
logger.Error(fmt.Errorf("Invalid password"), "")
|
|
break
|
|
}
|
|
|
|
if _, err := os.Stat("./history"); errors.Is(err, os.ErrNotExist) {
|
|
if err := os.Mkdir("./history", 0700); err != nil {
|
|
logger.Error(err, "Error creating directory")
|
|
break
|
|
}
|
|
}
|
|
|
|
var zeroscid crypto.Hash
|
|
account := wallet.GetAccount()
|
|
for k, v := range account.EntriesNative {
|
|
filename := filepath.Join("./history", k.String()+".json")
|
|
if k == zeroscid {
|
|
filename = filepath.Join("./history", "dero.json")
|
|
}
|
|
if data, err := json.Marshal(v); err != nil {
|
|
logger.Error(err, "Error exporting data")
|
|
} else if err = os.WriteFile(filename, data, 0600); err != nil {
|
|
logger.Error(err, "Error exporting data")
|
|
} else {
|
|
logger.Info("successfully exported history", "file", filename)
|
|
}
|
|
}
|
|
|
|
default:
|
|
processed = false // just loop
|
|
|
|
}
|
|
return
|
|
}
|