170 lines
5.1 KiB
Go
170 lines
5.1 KiB
Go
|
// Copyright 2018 The go-ethereum Authors
|
||
|
// This file is part of the go-ethereum library.
|
||
|
//
|
||
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
||
|
// it under the terms of the GNU Lesser General Public License as published by
|
||
|
// the Free Software Foundation, either version 3 of the License, or
|
||
|
// (at your option) any later version.
|
||
|
//
|
||
|
// The go-ethereum library is distributed in the hope that it will be useful,
|
||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
// GNU Lesser General Public License for more details.
|
||
|
//
|
||
|
// You should have received a copy of the GNU Lesser General Public License
|
||
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
||
|
|
||
|
package pss
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
|
||
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
||
|
"github.com/ethereum/go-ethereum/crypto"
|
||
|
"github.com/ethereum/go-ethereum/p2p"
|
||
|
"github.com/ethereum/go-ethereum/rpc"
|
||
|
"github.com/ethereum/go-ethereum/swarm/log"
|
||
|
)
|
||
|
|
||
|
// Wrapper for receiving pss messages when using the pss API
|
||
|
// providing access to sender of message
|
||
|
type APIMsg struct {
|
||
|
Msg hexutil.Bytes
|
||
|
Asymmetric bool
|
||
|
Key string
|
||
|
}
|
||
|
|
||
|
// Additional public methods accessible through API for pss
|
||
|
type API struct {
|
||
|
*Pss
|
||
|
}
|
||
|
|
||
|
func NewAPI(ps *Pss) *API {
|
||
|
return &API{Pss: ps}
|
||
|
}
|
||
|
|
||
|
// Creates a new subscription for the caller. Enables external handling of incoming messages.
|
||
|
//
|
||
|
// A new handler is registered in pss for the supplied topic
|
||
|
//
|
||
|
// All incoming messages to the node matching this topic will be encapsulated in the APIMsg
|
||
|
// struct and sent to the subscriber
|
||
|
func (pssapi *API) Receive(ctx context.Context, topic Topic) (*rpc.Subscription, error) {
|
||
|
notifier, supported := rpc.NotifierFromContext(ctx)
|
||
|
if !supported {
|
||
|
return nil, fmt.Errorf("Subscribe not supported")
|
||
|
}
|
||
|
|
||
|
psssub := notifier.CreateSubscription()
|
||
|
|
||
|
handler := func(msg []byte, p *p2p.Peer, asymmetric bool, keyid string) error {
|
||
|
apimsg := &APIMsg{
|
||
|
Msg: hexutil.Bytes(msg),
|
||
|
Asymmetric: asymmetric,
|
||
|
Key: keyid,
|
||
|
}
|
||
|
if err := notifier.Notify(psssub.ID, apimsg); err != nil {
|
||
|
log.Warn(fmt.Sprintf("notification on pss sub topic rpc (sub %v) msg %v failed!", psssub.ID, msg))
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
deregf := pssapi.Register(&topic, handler)
|
||
|
go func() {
|
||
|
defer deregf()
|
||
|
select {
|
||
|
case err := <-psssub.Err():
|
||
|
log.Warn(fmt.Sprintf("caught subscription error in pss sub topic %x: %v", topic, err))
|
||
|
case <-notifier.Closed():
|
||
|
log.Warn(fmt.Sprintf("rpc sub notifier closed"))
|
||
|
}
|
||
|
}()
|
||
|
|
||
|
return psssub, nil
|
||
|
}
|
||
|
|
||
|
func (pssapi *API) GetAddress(topic Topic, asymmetric bool, key string) (PssAddress, error) {
|
||
|
var addr *PssAddress
|
||
|
if asymmetric {
|
||
|
peer, ok := pssapi.Pss.pubKeyPool[key][topic]
|
||
|
if !ok {
|
||
|
return nil, fmt.Errorf("pubkey/topic pair %x/%x doesn't exist", key, topic)
|
||
|
}
|
||
|
addr = peer.address
|
||
|
} else {
|
||
|
peer, ok := pssapi.Pss.symKeyPool[key][topic]
|
||
|
if !ok {
|
||
|
return nil, fmt.Errorf("symkey/topic pair %x/%x doesn't exist", key, topic)
|
||
|
}
|
||
|
addr = peer.address
|
||
|
|
||
|
}
|
||
|
return *addr, nil
|
||
|
}
|
||
|
|
||
|
// Retrieves the node's base address in hex form
|
||
|
func (pssapi *API) BaseAddr() (PssAddress, error) {
|
||
|
return PssAddress(pssapi.Pss.BaseAddr()), nil
|
||
|
}
|
||
|
|
||
|
// Retrieves the node's public key in hex form
|
||
|
func (pssapi *API) GetPublicKey() (keybytes hexutil.Bytes) {
|
||
|
key := pssapi.Pss.PublicKey()
|
||
|
keybytes = crypto.FromECDSAPub(key)
|
||
|
return keybytes
|
||
|
}
|
||
|
|
||
|
// Set Public key to associate with a particular Pss peer
|
||
|
func (pssapi *API) SetPeerPublicKey(pubkey hexutil.Bytes, topic Topic, addr PssAddress) error {
|
||
|
pk, err := crypto.UnmarshalPubkey(pubkey)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("Cannot unmarshal pubkey: %x", pubkey)
|
||
|
}
|
||
|
err = pssapi.Pss.SetPeerPublicKey(pk, topic, &addr)
|
||
|
if err != nil {
|
||
|
return fmt.Errorf("Invalid key: %x", pk)
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (pssapi *API) GetSymmetricKey(symkeyid string) (hexutil.Bytes, error) {
|
||
|
symkey, err := pssapi.Pss.GetSymmetricKey(symkeyid)
|
||
|
return hexutil.Bytes(symkey), err
|
||
|
}
|
||
|
|
||
|
func (pssapi *API) GetSymmetricAddressHint(topic Topic, symkeyid string) (PssAddress, error) {
|
||
|
return *pssapi.Pss.symKeyPool[symkeyid][topic].address, nil
|
||
|
}
|
||
|
|
||
|
func (pssapi *API) GetAsymmetricAddressHint(topic Topic, pubkeyid string) (PssAddress, error) {
|
||
|
return *pssapi.Pss.pubKeyPool[pubkeyid][topic].address, nil
|
||
|
}
|
||
|
|
||
|
func (pssapi *API) StringToTopic(topicstring string) (Topic, error) {
|
||
|
topicbytes := BytesToTopic([]byte(topicstring))
|
||
|
if topicbytes == rawTopic {
|
||
|
return rawTopic, errors.New("Topic string hashes to 0x00000000 and cannot be used")
|
||
|
}
|
||
|
return topicbytes, nil
|
||
|
}
|
||
|
|
||
|
func (pssapi *API) SendAsym(pubkeyhex string, topic Topic, msg hexutil.Bytes) error {
|
||
|
return pssapi.Pss.SendAsym(pubkeyhex, topic, msg[:])
|
||
|
}
|
||
|
|
||
|
func (pssapi *API) SendSym(symkeyhex string, topic Topic, msg hexutil.Bytes) error {
|
||
|
return pssapi.Pss.SendSym(symkeyhex, topic, msg[:])
|
||
|
}
|
||
|
|
||
|
func (pssapi *API) GetPeerTopics(pubkeyhex string) ([]Topic, error) {
|
||
|
topics, _, err := pssapi.Pss.GetPublickeyPeers(pubkeyhex)
|
||
|
return topics, err
|
||
|
|
||
|
}
|
||
|
|
||
|
func (pssapi *API) GetPeerAddress(pubkeyhex string, topic Topic) (PssAddress, error) {
|
||
|
return pssapi.Pss.getPeerAddress(pubkeyhex, topic)
|
||
|
}
|