210 lines
5.1 KiB
Go
210 lines
5.1 KiB
Go
package wallet
|
|
|
|
import (
|
|
"context"
|
|
"math/big"
|
|
"time"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
"github.com/ethereum/go-ethereum/ethclient"
|
|
"github.com/ethereum/go-ethereum/event"
|
|
"github.com/ethereum/go-ethereum/log"
|
|
"github.com/ethereum/go-ethereum/p2p"
|
|
"github.com/ethereum/go-ethereum/rpc"
|
|
|
|
"github.com/status-im/status-go/multiaccounts/accounts"
|
|
)
|
|
|
|
// NewService initializes service instance.
|
|
func NewService(db *Database, accountsFeed *event.Feed) *Service {
|
|
feed := &event.Feed{}
|
|
return &Service{
|
|
db: db,
|
|
feed: feed,
|
|
signals: &SignalsTransmitter{
|
|
publisher: feed,
|
|
},
|
|
accountsFeed: accountsFeed,
|
|
}
|
|
}
|
|
|
|
// Service is a wallet service.
|
|
type Service struct {
|
|
feed *event.Feed
|
|
db *Database
|
|
reactor *Reactor
|
|
signals *SignalsTransmitter
|
|
client *ethclient.Client
|
|
cryptoOnRampManager *CryptoOnRampManager
|
|
started bool
|
|
|
|
group *Group
|
|
accountsFeed *event.Feed
|
|
}
|
|
|
|
// Start signals transmitter.
|
|
func (s *Service) Start(*p2p.Server) error {
|
|
s.group = NewGroup(context.Background())
|
|
return s.signals.Start()
|
|
}
|
|
|
|
// GetFeed returns signals feed.
|
|
func (s *Service) GetFeed() *event.Feed {
|
|
return s.feed
|
|
}
|
|
|
|
// SetClient sets ethclient
|
|
func (s *Service) SetClient(client *ethclient.Client) {
|
|
s.client = client
|
|
}
|
|
|
|
// MergeBlocksRanges merge old blocks ranges if possible
|
|
func (s *Service) MergeBlocksRanges(accounts []common.Address, chain uint64) error {
|
|
for _, account := range accounts {
|
|
err := s.db.mergeRanges(account, chain)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// StartReactor separately because it requires known ethereum address, which will become available only after login.
|
|
func (s *Service) StartReactor(client *ethclient.Client, accounts []common.Address, chain *big.Int, watchNewBlocks bool) error {
|
|
reactor := NewReactor(s.db, s.feed, client, chain, watchNewBlocks)
|
|
err := reactor.Start(accounts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
s.reactor = reactor
|
|
s.client = client
|
|
s.group.Add(func(ctx context.Context) error {
|
|
return WatchAccountsChanges(ctx, s.accountsFeed, accounts, reactor)
|
|
})
|
|
s.started = true
|
|
return nil
|
|
}
|
|
|
|
// StopReactor stops reactor and closes database.
|
|
func (s *Service) StopReactor() error {
|
|
if s.reactor == nil {
|
|
return nil
|
|
}
|
|
s.reactor.Stop()
|
|
if s.group != nil {
|
|
s.group.Stop()
|
|
s.group.Wait()
|
|
}
|
|
s.started = false
|
|
return nil
|
|
}
|
|
|
|
// Stop reactor, signals transmitter and close db.
|
|
func (s *Service) Stop() error {
|
|
log.Info("wallet will be stopped")
|
|
err := s.StopReactor()
|
|
s.signals.Stop()
|
|
if s.group != nil {
|
|
s.group.Stop()
|
|
s.group.Wait()
|
|
s.group = nil
|
|
}
|
|
log.Info("wallet stopped")
|
|
return err
|
|
}
|
|
|
|
// APIs returns list of available RPC APIs.
|
|
func (s *Service) APIs() []rpc.API {
|
|
return []rpc.API{
|
|
{
|
|
Namespace: "wallet",
|
|
Version: "0.1.0",
|
|
Service: NewAPI(s),
|
|
Public: true,
|
|
},
|
|
}
|
|
}
|
|
|
|
// Protocols returns list of p2p protocols.
|
|
func (s *Service) Protocols() []p2p.Protocol {
|
|
return nil
|
|
}
|
|
|
|
// WatchAccountsChanges subsribes to a feed and watches for changes in accounts list. If there are new or removed accounts
|
|
// reactor will be restarted.
|
|
func WatchAccountsChanges(ctx context.Context, feed *event.Feed, initial []common.Address, reactor *Reactor) error {
|
|
accounts := make(chan []accounts.Account, 1) // it may block if the rate of updates will be significantly higher
|
|
sub := feed.Subscribe(accounts)
|
|
defer sub.Unsubscribe()
|
|
listen := make(map[common.Address]struct{}, len(initial))
|
|
for _, address := range initial {
|
|
listen[address] = struct{}{}
|
|
}
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return nil
|
|
case err := <-sub.Err():
|
|
if err != nil {
|
|
log.Error("accounts watcher subscription failed", "error", err)
|
|
}
|
|
case n := <-accounts:
|
|
log.Debug("wallet received updated list of accounts", "accounts", n)
|
|
restart := false
|
|
for _, acc := range n {
|
|
_, exist := listen[common.Address(acc.Address)]
|
|
if !exist {
|
|
listen[common.Address(acc.Address)] = struct{}{}
|
|
restart = true
|
|
}
|
|
}
|
|
if !restart {
|
|
continue
|
|
}
|
|
listenList := mapToList(listen)
|
|
log.Debug("list of accounts was changed from a previous version. reactor will be restarted", "new", listenList)
|
|
reactor.Stop()
|
|
err := reactor.Start(listenList) // error is raised only if reactor is already running
|
|
if err != nil {
|
|
log.Error("failed to restart reactor with new accounts", "error", err)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func mapToList(m map[common.Address]struct{}) []common.Address {
|
|
rst := make([]common.Address, 0, len(m))
|
|
for address := range m {
|
|
rst = append(rst, address)
|
|
}
|
|
return rst
|
|
}
|
|
|
|
func (s *Service) IsStarted() bool {
|
|
return s.started
|
|
}
|
|
|
|
func (s *Service) SetInitialBlocksRange(network uint64) error {
|
|
accountsDB := accounts.NewDB(s.db.db)
|
|
watchAddress, err := accountsDB.GetWalletAddress()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
from := big.NewInt(0)
|
|
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
|
|
defer cancel()
|
|
|
|
header, err := s.client.HeaderByNumber(ctx, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = s.db.UpsertRange(common.Address(watchAddress), network, from, header.Number)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|