2022-06-15 15:49:31 +01:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
2022-06-15 16:36:17 +01:00
|
|
|
"crypto/ecdsa"
|
2022-08-31 12:44:12 +01:00
|
|
|
"crypto/elliptic"
|
2022-08-19 13:45:50 +01:00
|
|
|
"crypto/rand"
|
2022-06-15 15:49:31 +01:00
|
|
|
"crypto/tls"
|
2022-08-31 13:47:16 +01:00
|
|
|
"encoding/json"
|
2022-06-15 16:36:17 +01:00
|
|
|
"fmt"
|
|
|
|
"net"
|
2022-08-31 12:44:12 +01:00
|
|
|
"time"
|
2022-08-19 13:45:50 +01:00
|
|
|
|
|
|
|
"github.com/gorilla/sessions"
|
2022-08-31 13:47:16 +01:00
|
|
|
|
|
|
|
"github.com/status-im/status-go/multiaccounts"
|
2022-06-15 15:49:31 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type PairingServer struct {
|
|
|
|
Server
|
2022-07-05 06:40:43 +01:00
|
|
|
PayloadManager
|
2022-06-15 16:36:17 +01:00
|
|
|
|
2022-08-07 23:14:33 +01:00
|
|
|
pk *ecdsa.PublicKey
|
|
|
|
ek []byte
|
2022-07-05 06:40:43 +01:00
|
|
|
mode Mode
|
2022-08-19 13:45:50 +01:00
|
|
|
|
|
|
|
cookieStore *sessions.CookieStore
|
2022-06-15 15:49:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type Config struct {
|
2022-07-01 16:37:53 +01:00
|
|
|
// Connection fields
|
2022-08-07 23:14:33 +01:00
|
|
|
PK *ecdsa.PublicKey
|
|
|
|
EK []byte
|
2022-06-15 15:49:31 +01:00
|
|
|
Cert *tls.Certificate
|
|
|
|
Hostname string
|
2022-06-15 16:36:17 +01:00
|
|
|
Mode Mode
|
2022-07-01 16:37:53 +01:00
|
|
|
|
|
|
|
// Payload management fields
|
|
|
|
*PairingPayloadManagerConfig
|
2022-06-15 15:49:31 +01:00
|
|
|
}
|
|
|
|
|
2022-08-19 13:45:50 +01:00
|
|
|
func makeCookieStore() (*sessions.CookieStore, error) {
|
|
|
|
auth := make([]byte, 64)
|
|
|
|
_, err := rand.Read(auth)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
enc := make([]byte, 32)
|
|
|
|
_, err = rand.Read(enc)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return sessions.NewCookieStore(auth, enc), nil
|
|
|
|
}
|
|
|
|
|
2022-06-29 00:08:57 +01:00
|
|
|
// NewPairingServer returns a *PairingServer init from the given *Config
|
2022-07-01 16:37:53 +01:00
|
|
|
func NewPairingServer(config *Config) (*PairingServer, error) {
|
2022-08-07 23:14:33 +01:00
|
|
|
pm, err := NewPairingPayloadManager(config.EK, config.PairingPayloadManagerConfig)
|
2022-06-10 16:32:15 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-08-19 13:45:50 +01:00
|
|
|
cs, err := makeCookieStore()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-06-15 15:49:31 +01:00
|
|
|
return &PairingServer{Server: NewServer(
|
|
|
|
config.Cert,
|
|
|
|
config.Hostname,
|
2022-08-07 15:35:54 +08:00
|
|
|
nil,
|
2022-06-15 16:36:17 +01:00
|
|
|
),
|
2022-07-01 16:37:53 +01:00
|
|
|
pk: config.PK,
|
2022-08-07 23:14:33 +01:00
|
|
|
ek: config.EK,
|
2022-07-01 16:37:53 +01:00
|
|
|
mode: config.Mode,
|
2022-08-19 13:45:50 +01:00
|
|
|
PayloadManager: pm,
|
|
|
|
cookieStore: cs,
|
|
|
|
}, nil
|
2022-06-15 16:36:17 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// MakeConnectionParams generates a *ConnectionParams based on the Server's current state
|
|
|
|
func (s *PairingServer) MakeConnectionParams() (*ConnectionParams, error) {
|
|
|
|
netIP := net.ParseIP(s.hostname)
|
|
|
|
if netIP == nil {
|
|
|
|
return nil, fmt.Errorf("invalid ip address given '%s'", s.hostname)
|
|
|
|
}
|
|
|
|
|
|
|
|
netIP4 := netIP.To4()
|
|
|
|
if netIP4 != nil {
|
|
|
|
netIP = netIP4
|
|
|
|
}
|
|
|
|
|
|
|
|
if s.port == 0 {
|
|
|
|
return nil, fmt.Errorf("port is 0, listener is not yet set")
|
|
|
|
}
|
|
|
|
|
2022-08-07 23:14:33 +01:00
|
|
|
return NewConnectionParams(netIP, s.port, s.pk, s.ek, s.mode), nil
|
2022-06-15 15:49:31 +01:00
|
|
|
}
|
2022-06-10 16:32:15 +01:00
|
|
|
|
|
|
|
func (s *PairingServer) StartPairing() error {
|
|
|
|
switch s.mode {
|
|
|
|
case Receiving:
|
|
|
|
return s.startReceivingAccountData()
|
|
|
|
case Sending:
|
|
|
|
return s.startSendingAccountData()
|
|
|
|
default:
|
|
|
|
return fmt.Errorf("invalid server mode '%d'", s.mode)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *PairingServer) startReceivingAccountData() error {
|
2022-08-19 13:45:50 +01:00
|
|
|
s.SetHandlers(HandlerPatternMap{
|
|
|
|
pairingReceive: handlePairingReceive(s),
|
|
|
|
pairingChallenge: handlePairingChallenge(s),
|
|
|
|
})
|
2022-06-10 16:32:15 +01:00
|
|
|
return s.Start()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *PairingServer) startSendingAccountData() error {
|
2022-08-31 13:47:16 +01:00
|
|
|
err := s.Mount()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-08-19 13:45:50 +01:00
|
|
|
s.SetHandlers(HandlerPatternMap{
|
|
|
|
pairingSend: challengeMiddleware(s, handlePairingSend(s)),
|
|
|
|
pairingChallenge: handlePairingChallenge(s),
|
|
|
|
})
|
2022-06-10 16:32:15 +01:00
|
|
|
return s.Start()
|
|
|
|
}
|
2022-08-31 12:44:12 +01:00
|
|
|
|
2022-08-31 15:01:45 +01:00
|
|
|
// MakeFullPairingServer generates a fully configured and randomly seeded PairingServer
|
2022-08-31 12:58:59 +01:00
|
|
|
func MakeFullPairingServer(db *multiaccounts.Database, mode Mode, storeConfig PairingPayloadSourceConfig) (*PairingServer, error) {
|
2022-08-31 12:44:12 +01:00
|
|
|
tlsKey, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
AESKey := make([]byte, 32)
|
|
|
|
_, err = rand.Read(AESKey)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
outboundIP, err := GetOutboundIP()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
tlsCert, _, err := GenerateCertFromKey(tlsKey, time.Now(), outboundIP.String())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return NewPairingServer(&Config{
|
2022-08-31 12:58:59 +01:00
|
|
|
// Things that can be generated, and CANNOT come from the app client (well they could be this is better)
|
2022-08-31 12:44:12 +01:00
|
|
|
PK: &tlsKey.PublicKey,
|
|
|
|
EK: AESKey,
|
|
|
|
Cert: &tlsCert,
|
|
|
|
Hostname: outboundIP.String(),
|
|
|
|
|
2022-08-31 12:58:59 +01:00
|
|
|
// Things that can't be generated, but DO come from the app client
|
2022-08-31 12:44:12 +01:00
|
|
|
Mode: mode,
|
|
|
|
|
|
|
|
PairingPayloadManagerConfig: &PairingPayloadManagerConfig{
|
2022-08-31 12:58:59 +01:00
|
|
|
// Things that can't be generated, but DO NOT come from app client
|
2022-08-31 12:44:12 +01:00
|
|
|
DB: db,
|
|
|
|
|
2022-08-31 12:58:59 +01:00
|
|
|
// Things that can't be generated, but DO come from the app client
|
|
|
|
PairingPayloadSourceConfig: storeConfig,
|
2022-08-31 12:44:12 +01:00
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
2022-08-31 13:47:16 +01:00
|
|
|
|
2022-08-31 15:01:45 +01:00
|
|
|
// StartUpPairingServer generates a PairingServer, starts the pairing server in the correct mode
|
|
|
|
// and returns the ConnectionParams string to allow a PairingClient to make a successful connection.
|
2022-08-31 13:47:16 +01:00
|
|
|
func StartUpPairingServer(db *multiaccounts.Database, mode Mode, configJSON string) (string, error) {
|
|
|
|
var conf PairingPayloadSourceConfig
|
|
|
|
err := json.Unmarshal([]byte(configJSON), &conf)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
ps, err := MakeFullPairingServer(db, mode, conf)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = ps.StartPairing()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
cp, err := ps.MakeConnectionParams()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return cp.ToString(), nil
|
|
|
|
}
|