mirror of
https://github.com/status-im/keycard-go.git
synced 2025-01-09 19:36:36 +00:00
57 lines
1.3 KiB
Go
57 lines
1.3 KiB
Go
package lightwallet
|
|
|
|
import (
|
|
"crypto/ecdsa"
|
|
|
|
ethcrypto "github.com/ethereum/go-ethereum/crypto"
|
|
"github.com/status-im/smartcard-go/apdu"
|
|
"github.com/status-im/smartcard-go/globalplatform"
|
|
"github.com/status-im/smartcard-go/lightwallet/crypto"
|
|
)
|
|
|
|
type SecureChannel struct {
|
|
c globalplatform.Channel
|
|
secret []byte
|
|
publicKey *ecdsa.PublicKey
|
|
}
|
|
|
|
func NewSecureChannel(c globalplatform.Channel, cardKeyData []byte) (*SecureChannel, error) {
|
|
key, err := ethcrypto.GenerateKey()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
cardPubKey, err := ethcrypto.UnmarshalPubkey(cardKeyData)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
secret := crypto.GenerateECDHSharedSecret(key, cardPubKey)
|
|
|
|
return &SecureChannel{
|
|
c: c,
|
|
secret: secret,
|
|
publicKey: &key.PublicKey,
|
|
}, nil
|
|
}
|
|
|
|
func (sc *SecureChannel) PublicKey() *ecdsa.PublicKey {
|
|
return sc.publicKey
|
|
}
|
|
|
|
func (sc *SecureChannel) RawPublicKey() []byte {
|
|
return ethcrypto.FromECDSAPub(sc.publicKey)
|
|
}
|
|
|
|
func (sc *SecureChannel) Send(cmd *apdu.Command) (*apdu.Response, error) {
|
|
return sc.c.Send(cmd)
|
|
}
|
|
|
|
func (sc *SecureChannel) OneShotEncrypt(secrets *Secrets) ([]byte, error) {
|
|
pubKeyData := ethcrypto.FromECDSAPub(sc.publicKey)
|
|
data := append([]byte(secrets.Pin()), []byte(secrets.Puk())...)
|
|
data = append(data, secrets.PairingToken()...)
|
|
|
|
return crypto.OneShotEncrypt(pubKeyData, sc.secret, data)
|
|
}
|