mirror of
https://github.com/status-im/status-go.git
synced 2025-01-12 07:35:02 +00:00
f04a9a8726
* feat(sync)!: remove compatibility with v2.29 * feat(sync)_: add AC notifications when initiating the sync fallback Needed for https://github.com/status-im/status-desktop/issues/15750 Adds an AC notification when the syncing fails and the user is prompted to use a seed phrase instead. There is one notification for the initiator (created) and one for the old account (received). Once the flow is completed, ie the receiver presses Enable and sync, the notifications are deleted * test_: update test * fix_: lint issue * chore_: ignore tmp file generated by make lint-fix * chore_: rename EnableAndSyncInstallation to EnableInstallationAndSync * chore_: address review feedback * chore_: revert changes to .gitignore * fix_: simplify code * fix_: keep old API --------- Co-authored-by: Jonathan Rainville <rainville.jonathan@gmail.com>
283 lines
7.2 KiB
Go
283 lines
7.2 KiB
Go
package pairing
|
|
|
|
import (
|
|
"crypto/ecdsa"
|
|
"crypto/elliptic"
|
|
"fmt"
|
|
"log"
|
|
"math/big"
|
|
"net"
|
|
"net/url"
|
|
"strings"
|
|
|
|
"github.com/btcsuite/btcutil/base58"
|
|
"github.com/google/uuid"
|
|
|
|
"github.com/status-im/status-go/server/pairing/versioning"
|
|
)
|
|
|
|
const (
|
|
connectionStringID = "cs"
|
|
)
|
|
|
|
type ConnectionParams struct {
|
|
version versioning.ConnectionParamVersion
|
|
netIPs []net.IP
|
|
port int
|
|
publicKey *ecdsa.PublicKey
|
|
aesKey []byte
|
|
installationID string
|
|
keyUID string
|
|
}
|
|
|
|
func NewConnectionParams(netIPs []net.IP, port int, publicKey *ecdsa.PublicKey, aesKey []byte, installationID, keyUID string) *ConnectionParams {
|
|
cp := new(ConnectionParams)
|
|
cp.version = versioning.LatestConnectionParamVer
|
|
cp.netIPs = netIPs
|
|
cp.port = port
|
|
cp.publicKey = publicKey
|
|
cp.aesKey = aesKey
|
|
cp.installationID = installationID
|
|
cp.keyUID = keyUID
|
|
return cp
|
|
}
|
|
|
|
// ToString generates a string required for generating a secure connection to another Status device.
|
|
//
|
|
// The returned string will look like below:
|
|
// - "cs2:4FHRnp:H6G:uqnnMwVUfJc2Fkcaojet8F1ufKC3hZdGEt47joyBx9yd:BbnZ7Gc66t54a9kEFCf7FW8SGQuYypwHVeNkRYeNoqV6"
|
|
//
|
|
// Format bytes encoded into a base58 string, delimited by ":"
|
|
// - string type identifier
|
|
// - version
|
|
// - net.IP
|
|
// - array of IPs in next form:
|
|
// | 1 byte | 4*N bytes | 1 byte | 16*N bytes |
|
|
// | N | N * IPv4 | M | M * IPv6 |
|
|
// - port
|
|
// - ecdsa CompressedPublicKey
|
|
// - AES encryption key
|
|
// - string InstallationID of the sending device
|
|
// - string KeyUID of the sending device
|
|
//
|
|
// NOTE:
|
|
// - append(accrete) parameters instead of changing(breaking) existing parameters. Appending should **never** break, modifying existing parameters will break. Watch this before making changes: https://www.youtube.com/watch?v=oyLBGkS5ICk
|
|
// - never strictly check version, unless you really want to break
|
|
func (cp *ConnectionParams) ToString() string {
|
|
v := base58.Encode(new(big.Int).SetInt64(int64(cp.version)).Bytes())
|
|
ips := base58.Encode(SerializeNetIps(cp.netIPs))
|
|
p := base58.Encode(new(big.Int).SetInt64(int64(cp.port)).Bytes())
|
|
k := base58.Encode(elliptic.MarshalCompressed(cp.publicKey.Curve, cp.publicKey.X, cp.publicKey.Y))
|
|
ek := base58.Encode(cp.aesKey)
|
|
|
|
var i string
|
|
if cp.installationID != "" {
|
|
u, err := uuid.Parse(cp.installationID)
|
|
if err != nil {
|
|
log.Fatalf("Failed to parse UUID: %v", err)
|
|
} else {
|
|
// Convert UUID to byte slice
|
|
byteSlice := u[:]
|
|
i = base58.Encode(byteSlice)
|
|
}
|
|
}
|
|
|
|
var kuid string
|
|
if cp.keyUID != "" {
|
|
kuid = base58.Encode([]byte(cp.keyUID))
|
|
}
|
|
|
|
return fmt.Sprintf("%s%s:%s:%s:%s:%s:%s:%s", connectionStringID, v, ips, p, k, ek, i, kuid)
|
|
}
|
|
|
|
func (cp *ConnectionParams) InstallationID() string {
|
|
return cp.installationID
|
|
}
|
|
|
|
func (cp *ConnectionParams) KeyUID() string {
|
|
return cp.keyUID
|
|
}
|
|
|
|
func SerializeNetIps(ips []net.IP) []byte {
|
|
var out []byte
|
|
var ipv4 []net.IP
|
|
var ipv6 []net.IP
|
|
|
|
for _, ip := range ips {
|
|
if v := ip.To4(); v != nil {
|
|
ipv4 = append(ipv4, v)
|
|
} else {
|
|
ipv6 = append(ipv6, ip)
|
|
}
|
|
}
|
|
|
|
for _, arr := range [][]net.IP{ipv4, ipv6} {
|
|
out = append(out, uint8(len(arr)))
|
|
for _, ip := range arr {
|
|
out = append(out, ip...)
|
|
}
|
|
}
|
|
|
|
return out
|
|
}
|
|
|
|
func ParseNetIps(in []byte) ([]net.IP, error) {
|
|
var out []net.IP
|
|
|
|
if len(in) < 1 {
|
|
return nil, fmt.Errorf("net.ip field is too short: '%d', at least 1 byte required", len(in))
|
|
}
|
|
|
|
for _, ipLen := range []int{net.IPv4len, net.IPv6len} {
|
|
|
|
count := int(in[0])
|
|
in = in[1:]
|
|
|
|
if expectedLen := ipLen * count; len(in) < expectedLen {
|
|
return nil, fmt.Errorf("net.ip.ip%d field is too short, expected at least '%d' bytes, '%d' bytes found", ipLen, expectedLen, len(in))
|
|
}
|
|
|
|
for i := 0; i < count; i++ {
|
|
offset := i * ipLen
|
|
ip := in[offset : ipLen+offset]
|
|
out = append(out, ip)
|
|
}
|
|
|
|
in = in[ipLen*count:]
|
|
}
|
|
|
|
return out, nil
|
|
}
|
|
|
|
// FromString parses a connection params string required for to securely connect to another Status device.
|
|
// This function parses a connection string generated by ToString
|
|
func (cp *ConnectionParams) FromString(s string) error {
|
|
|
|
if len(s) < 2 {
|
|
return fmt.Errorf("connection string is too short: '%s'", s)
|
|
}
|
|
|
|
if s[:2] != connectionStringID {
|
|
return fmt.Errorf("connection string doesn't begin with identifier '%s'", connectionStringID)
|
|
}
|
|
|
|
requiredParams := 5
|
|
|
|
sData := strings.Split(s[2:], ":")
|
|
// NOTE: always allow extra parameters for forward compatibility, error on not enough required parameters or failing to parse
|
|
if len(sData) < requiredParams {
|
|
return fmt.Errorf("expected data '%s' to have length of '%d', received '%d'", s, requiredParams, len(sData))
|
|
}
|
|
|
|
netIpsBytes := base58.Decode(sData[1])
|
|
netIps, err := ParseNetIps(netIpsBytes)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
cp.netIPs = netIps
|
|
|
|
cp.port = int(new(big.Int).SetBytes(base58.Decode(sData[2])).Int64())
|
|
cp.publicKey = new(ecdsa.PublicKey)
|
|
cp.publicKey.X, cp.publicKey.Y = elliptic.UnmarshalCompressed(elliptic.P256(), base58.Decode(sData[3]))
|
|
cp.publicKey.Curve = elliptic.P256()
|
|
cp.aesKey = base58.Decode(sData[4])
|
|
|
|
if len(sData) > 5 && len(sData[5]) != 0 {
|
|
installationIDBytes := base58.Decode(sData[5])
|
|
installationID, err := uuid.FromBytes(installationIDBytes)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
cp.installationID = installationID.String()
|
|
}
|
|
|
|
if len(sData) > 6 && len(sData[6]) != 0 {
|
|
decodedBytes := base58.Decode(sData[6])
|
|
cp.keyUID = string(decodedBytes)
|
|
}
|
|
|
|
return cp.validate()
|
|
}
|
|
|
|
func (cp *ConnectionParams) validate() error {
|
|
err := cp.validateNetIP()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = cp.validatePort()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = cp.validatePublicKey()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return cp.validateAESKey()
|
|
}
|
|
|
|
func (cp *ConnectionParams) validateNetIP() error {
|
|
for _, ip := range cp.netIPs {
|
|
if ok := net.ParseIP(ip.String()); ok == nil {
|
|
return fmt.Errorf("invalid net ip '%s'", cp.netIPs)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (cp *ConnectionParams) validatePort() error {
|
|
if cp.port > 0 && cp.port < 0x10000 {
|
|
return nil
|
|
}
|
|
|
|
return fmt.Errorf("port '%d' outside of bounds of 1 - 65535", cp.port)
|
|
}
|
|
|
|
func (cp *ConnectionParams) validatePublicKey() error {
|
|
switch {
|
|
case cp.publicKey.Curve == nil, cp.publicKey.Curve != elliptic.P256():
|
|
return fmt.Errorf("public key Curve not `elliptic.P256`")
|
|
case cp.publicKey.X == nil, cp.publicKey.X.Cmp(big.NewInt(0)) == 0:
|
|
return fmt.Errorf("public key X not set")
|
|
case cp.publicKey.Y == nil, cp.publicKey.Y.Cmp(big.NewInt(0)) == 0:
|
|
return fmt.Errorf("public key Y not set")
|
|
default:
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func (cp *ConnectionParams) validateAESKey() error {
|
|
if len(cp.aesKey) != 32 {
|
|
return fmt.Errorf("AES key invalid length, expect length 32, received length '%d'", len(cp.aesKey))
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (cp *ConnectionParams) URL(IPIndex int) (*url.URL, error) {
|
|
if IPIndex < 0 || IPIndex >= len(cp.netIPs) {
|
|
return nil, fmt.Errorf("invalid IP index '%d'", IPIndex)
|
|
}
|
|
|
|
err := cp.validate()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return cp.BuildURL(cp.netIPs[IPIndex]), nil
|
|
}
|
|
|
|
func (cp *ConnectionParams) BuildURL(ip net.IP) *url.URL {
|
|
return &url.URL{
|
|
Scheme: "https",
|
|
Host: fmt.Sprintf("%s:%d", ip, cp.port),
|
|
}
|
|
}
|
|
|
|
func ValidateConnectionString(cs string) error {
|
|
ccp := ConnectionParams{}
|
|
err := ccp.FromString(cs)
|
|
return err
|
|
}
|