status-keycard-go/pkg/flow/mocked/mocked_flow_export_public.go
2025-01-20 11:57:27 +01:00

183 lines
5.4 KiB
Go

package mocked
import (
"math/rand"
"strconv"
"strings"
"github.com/status-im/status-keycard-go/internal"
"github.com/status-im/status-keycard-go/pkg/flow"
"github.com/status-im/status-keycard-go/signal"
)
func (mkf *MockedKeycardFlow) handleExportPublicFlow() {
flowStatus := flow.FlowStatus{}
if mkf.insertedKeycard.NotStatusKeycard {
flowStatus[internal.ErrorKey] = internal.ErrorNotAKeycard
flowStatus[flow.InstanceUID] = ""
flowStatus[flow.KeyUID] = ""
flowStatus[flow.FreeSlots] = 0
mkf.state = flow.Paused
signal.Send(flow.SwapCard, flowStatus)
return
}
flowStatus = flow.FlowStatus{
flow.InstanceUID: mkf.insertedKeycard.InstanceUID,
flow.KeyUID: mkf.insertedKeycard.KeyUID,
}
if mkf.insertedKeycard.InstanceUID == "" || mkf.insertedKeycard.KeyUID == "" {
flowStatus[internal.ErrorKey] = internal.ErrorNoKeys
flowStatus[flow.FreeSlots] = 0
mkf.state = flow.Paused
signal.Send(flow.SwapCard, flowStatus)
return
}
var (
enteredPIN string
enteredNewPIN string
enteredPUK string
exportMaster bool
exportPrivate bool
)
if v, ok := mkf.params[flow.PIN]; ok {
enteredPIN = v.(string)
}
if v, ok := mkf.params[flow.NewPIN]; ok {
enteredNewPIN = v.(string)
}
if v, ok := mkf.params[flow.PUK]; ok {
enteredPUK = v.(string)
}
if v, ok := mkf.params[flow.ExportMaster]; ok {
exportMaster = v.(bool)
}
if v, ok := mkf.params[flow.ExportPriv]; ok {
exportPrivate = v.(bool)
}
finalType := flow.EnterPIN
if mkf.insertedKeycard.PukRetries == 0 {
flowStatus[internal.ErrorKey] = flow.PUKRetries
finalType = flow.SwapCard
} else {
if mkf.insertedKeycard.PinRetries == 0 {
if len(enteredPUK) == internal.DefPUKLen {
if len(enteredPIN) == internal.DefPINLen && enteredPIN == enteredNewPIN {
if enteredPUK != mkf.insertedKeycard.Puk {
mkf.insertedKeycard.PukRetries--
if mkf.insertedKeycard.PukRetries == 0 {
flowStatus[internal.ErrorKey] = flow.PUKRetries
finalType = flow.SwapCard
} else {
flowStatus[internal.ErrorKey] = flow.PUK
finalType = flow.EnterPUK
}
}
} else {
flowStatus[internal.ErrorKey] = internal.ErrorUnblocking
finalType = flow.EnterNewPIN
}
} else {
flowStatus[internal.ErrorKey] = ""
finalType = flow.EnterPUK
}
} else {
if len(enteredNewPIN) == 0 && len(enteredPIN) == internal.DefPINLen && enteredPIN != mkf.insertedKeycard.Pin {
mkf.insertedKeycard.PinRetries--
flowStatus[internal.ErrorKey] = flow.PIN
finalType = flow.EnterPIN
if mkf.insertedKeycard.PinRetries == 0 {
flowStatus[internal.ErrorKey] = ""
finalType = flow.EnterPUK
}
}
}
}
if mkf.insertedKeycard.PinRetries > 0 && len(enteredPIN) == internal.DefPINLen && enteredPIN == mkf.insertedKeycard.Pin ||
mkf.insertedKeycard.PinRetries == 0 && mkf.insertedKeycard.PukRetries > 0 && len(enteredPUK) == internal.DefPUKLen &&
enteredPUK == mkf.insertedKeycard.Puk && len(enteredPIN) == internal.DefPINLen && enteredPIN == enteredNewPIN {
mkf.insertedKeycard.PinRetries = internal.MaxPINRetries
mkf.insertedKeycard.PukRetries = internal.MaxPUKRetries
mkf.insertedKeycard.Pin = enteredPIN
if exportMaster {
if mkf.insertedKeycardHelper.MasterKeyAddress == "" {
iAsStr := strconv.Itoa(rand.Intn(100) + 100)
mkf.insertedKeycardHelper.MasterKeyAddress = "0x" + strings.Repeat("0", 40-len(iAsStr)) + iAsStr
}
flowStatus[flow.MasterAddr] = mkf.insertedKeycardHelper.MasterKeyAddress
}
if path, ok := mkf.params[flow.BIP44Path]; ok {
if mkf.insertedKeycardHelper.ExportedKey == nil {
mkf.insertedKeycardHelper.ExportedKey = make(map[string]internal.KeyPair)
}
if pathStr, ok := path.(string); ok {
keyPair := mkf.insertedKeycardHelper.ExportedKey[pathStr]
if keyPair.Address == "" {
keyPair.Address = "0x" + strings.Repeat("0", 39) + "1"
}
if len(keyPair.PublicKey) == 0 {
keyPair.PublicKey = []byte(strings.Repeat("0", 129) + "1")
}
if !exportPrivate {
keyPair.PrivateKey = []byte("")
} else if len(keyPair.PrivateKey) == 0 {
keyPair.PrivateKey = []byte(strings.Repeat("0", 63) + "1")
}
mkf.insertedKeycardHelper.ExportedKey[pathStr] = keyPair
flowStatus[flow.ExportedKey] = keyPair
} else if paths, ok := path.([]interface{}); ok {
keys := make([]*internal.KeyPair, len(paths))
for i, path := range paths {
keyPair := mkf.insertedKeycardHelper.ExportedKey[path.(string)]
if keyPair.Address == "" {
iAsStr := strconv.Itoa(i + 1)
keyPair.Address = "0x" + strings.Repeat("0", 40-len(iAsStr)) + iAsStr
}
if len(keyPair.PublicKey) == 0 {
iAsStr := strconv.Itoa(i + 1)
keyPair.PublicKey = []byte(strings.Repeat("0", 130-len(iAsStr)) + iAsStr)
}
if !exportPrivate {
keyPair.PrivateKey = []byte("")
} else if len(keyPair.PrivateKey) == 0 {
iAsStr := strconv.Itoa(i + 1)
keyPair.PrivateKey = []byte(strings.Repeat("0", 64-len(iAsStr)) + iAsStr)
}
mkf.insertedKeycardHelper.ExportedKey[path.(string)] = keyPair
keys[i] = &keyPair
}
flowStatus[flow.ExportedKey] = keys
}
}
mkf.state = flow.Idle
signal.Send(flow.FlowResult, flowStatus)
return
}
flowStatus[flow.FreeSlots] = mkf.insertedKeycard.FreePairingSlots
flowStatus[flow.PINRetries] = mkf.insertedKeycard.PinRetries
flowStatus[flow.PUKRetries] = mkf.insertedKeycard.PukRetries
mkf.state = flow.Paused
signal.Send(finalType, flowStatus)
}