2017-05-16 15:09:52 +03:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
2017-09-04 14:56:58 +02:00
|
|
|
"context"
|
|
|
|
|
2017-10-23 18:46:51 +03:00
|
|
|
"github.com/NaySoftware/go-fcm"
|
2017-05-16 15:09:52 +03:00
|
|
|
"github.com/ethereum/go-ethereum/accounts/keystore"
|
|
|
|
gethcommon "github.com/ethereum/go-ethereum/common"
|
2018-03-20 14:35:28 -04:00
|
|
|
"github.com/ethereum/go-ethereum/log"
|
2018-03-22 13:31:12 +01:00
|
|
|
"github.com/status-im/status-go/geth/account"
|
2017-05-16 15:09:52 +03:00
|
|
|
"github.com/status-im/status-go/geth/common"
|
2018-03-01 11:48:30 -05:00
|
|
|
"github.com/status-im/status-go/geth/jail"
|
2017-05-16 15:09:52 +03:00
|
|
|
"github.com/status-im/status-go/geth/params"
|
2018-01-22 11:31:24 +02:00
|
|
|
"github.com/status-im/status-go/geth/transactions"
|
2017-05-16 15:09:52 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
// StatusAPI provides API to access Status related functionality.
|
|
|
|
type StatusAPI struct {
|
2018-03-20 14:35:28 -04:00
|
|
|
b *StatusBackend
|
|
|
|
log log.Logger
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
2017-12-02 19:51:55 +01:00
|
|
|
// NewStatusAPI creates a new StatusAPI instance
|
2017-05-16 15:09:52 +03:00
|
|
|
func NewStatusAPI() *StatusAPI {
|
2017-12-02 19:51:55 +01:00
|
|
|
return NewStatusAPIWithBackend(NewStatusBackend())
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewStatusAPIWithBackend creates a new StatusAPI instance using
|
|
|
|
// the passed backend.
|
|
|
|
func NewStatusAPIWithBackend(b *StatusBackend) *StatusAPI {
|
2017-05-16 15:09:52 +03:00
|
|
|
return &StatusAPI{
|
2018-03-20 14:35:28 -04:00
|
|
|
b: b,
|
|
|
|
log: log.New("package", "status-go/geth/api.StatusAPI"),
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NodeManager returns reference to node manager
|
|
|
|
func (api *StatusAPI) NodeManager() common.NodeManager {
|
|
|
|
return api.b.NodeManager()
|
|
|
|
}
|
|
|
|
|
|
|
|
// AccountManager returns reference to account manager
|
2018-03-22 13:31:12 +01:00
|
|
|
func (api *StatusAPI) AccountManager() *account.Manager {
|
2017-05-16 15:09:52 +03:00
|
|
|
return api.b.AccountManager()
|
|
|
|
}
|
|
|
|
|
|
|
|
// JailManager returns reference to jail
|
2018-03-01 11:48:30 -05:00
|
|
|
func (api *StatusAPI) JailManager() jail.Manager {
|
2017-05-16 15:09:52 +03:00
|
|
|
return api.b.JailManager()
|
|
|
|
}
|
|
|
|
|
2017-09-04 14:56:58 +02:00
|
|
|
// TxQueueManager returns reference to account manager
|
2018-01-22 11:31:24 +02:00
|
|
|
func (api *StatusAPI) TxQueueManager() *transactions.Manager {
|
2017-09-04 14:56:58 +02:00
|
|
|
return api.b.TxQueueManager()
|
|
|
|
}
|
|
|
|
|
2017-05-16 15:09:52 +03:00
|
|
|
// StartNode start Status node, fails if node is already started
|
|
|
|
func (api *StatusAPI) StartNode(config *params.NodeConfig) error {
|
2018-02-09 15:37:56 +02:00
|
|
|
return api.b.StartNode(config)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
2017-05-25 16:14:52 +03:00
|
|
|
// StartNodeAsync start Status node, fails if node is already started
|
|
|
|
// Returns immediately w/o waiting for node to start (see node.ready)
|
2018-02-09 15:37:56 +02:00
|
|
|
func (api *StatusAPI) StartNodeAsync(config *params.NodeConfig) <-chan error {
|
|
|
|
return runAsync(func() error { return api.StartNode(config) })
|
2017-05-25 16:14:52 +03:00
|
|
|
}
|
2017-05-25 15:34:13 +03:00
|
|
|
|
2017-05-25 16:14:52 +03:00
|
|
|
// StopNode stop Status node. Stopped node cannot be resumed.
|
|
|
|
func (api *StatusAPI) StopNode() error {
|
2018-02-09 15:37:56 +02:00
|
|
|
return api.b.StopNode()
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
2017-05-25 16:14:52 +03:00
|
|
|
// StopNodeAsync stop Status node. Stopped node cannot be resumed.
|
|
|
|
// Returns immediately, w/o waiting for node to stop (see node.stopped)
|
2018-02-09 15:37:56 +02:00
|
|
|
func (api *StatusAPI) StopNodeAsync() <-chan error {
|
|
|
|
return runAsync(api.StopNode)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// RestartNode restart running Status node, fails if node is not running
|
|
|
|
func (api *StatusAPI) RestartNode() error {
|
2018-02-09 15:37:56 +02:00
|
|
|
return api.b.RestartNode()
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
2017-05-25 16:14:52 +03:00
|
|
|
// RestartNodeAsync restart running Status node, in async manner
|
2018-02-09 15:37:56 +02:00
|
|
|
func (api *StatusAPI) RestartNodeAsync() <-chan error {
|
|
|
|
return runAsync(api.RestartNode)
|
2017-05-25 16:14:52 +03:00
|
|
|
}
|
|
|
|
|
2017-05-16 15:09:52 +03:00
|
|
|
// ResetChainData remove chain data from data directory.
|
|
|
|
// Node is stopped, and new node is started, with clean data directory.
|
|
|
|
func (api *StatusAPI) ResetChainData() error {
|
2018-02-09 15:37:56 +02:00
|
|
|
return api.b.ResetChainData()
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
2017-05-25 16:14:52 +03:00
|
|
|
// ResetChainDataAsync remove chain data from data directory, in async manner
|
2018-02-09 15:37:56 +02:00
|
|
|
func (api *StatusAPI) ResetChainDataAsync() <-chan error {
|
|
|
|
return runAsync(api.ResetChainData)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
2017-05-28 16:57:30 +03:00
|
|
|
// CallRPC executes RPC request on node's in-proc RPC server
|
|
|
|
func (api *StatusAPI) CallRPC(inputJSON string) string {
|
|
|
|
return api.b.CallRPC(inputJSON)
|
|
|
|
}
|
|
|
|
|
2017-05-16 15:09:52 +03:00
|
|
|
// CreateAccount creates an internal geth account
|
|
|
|
// BIP44-compatible keys are generated: CKD#1 is stored as account key, CKD#2 stored as sub-account root
|
|
|
|
// Public key of CKD#1 is returned, with CKD#2 securely encoded into account key file (to be used for
|
|
|
|
// sub-account derivations)
|
|
|
|
func (api *StatusAPI) CreateAccount(password string) (address, pubKey, mnemonic string, err error) {
|
2017-05-25 16:14:52 +03:00
|
|
|
return api.b.AccountManager().CreateAccount(password)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// CreateChildAccount creates sub-account for an account identified by parent address.
|
|
|
|
// CKD#2 is used as root for master accounts (when parentAddress is "").
|
|
|
|
// Otherwise (when parentAddress != ""), child is derived directly from parent.
|
|
|
|
func (api *StatusAPI) CreateChildAccount(parentAddress, password string) (address, pubKey string, err error) {
|
2017-05-25 16:14:52 +03:00
|
|
|
return api.b.AccountManager().CreateChildAccount(parentAddress, password)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// RecoverAccount re-creates master key using given details.
|
|
|
|
// Once master key is re-generated, it is inserted into keystore (if not already there).
|
|
|
|
func (api *StatusAPI) RecoverAccount(password, mnemonic string) (address, pubKey string, err error) {
|
2017-05-25 16:14:52 +03:00
|
|
|
return api.b.AccountManager().RecoverAccount(password, mnemonic)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// VerifyAccountPassword tries to decrypt a given account key file, with a provided password.
|
|
|
|
// If no error is returned, then account is considered verified.
|
|
|
|
func (api *StatusAPI) VerifyAccountPassword(keyStoreDir, address, password string) (*keystore.Key, error) {
|
2017-05-25 16:14:52 +03:00
|
|
|
return api.b.AccountManager().VerifyAccountPassword(keyStoreDir, address, password)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// SelectAccount selects current account, by verifying that address has corresponding account which can be decrypted
|
|
|
|
// using provided password. Once verification is done, decrypted key is injected into Whisper (as a single identity,
|
|
|
|
// all previous identities are removed).
|
|
|
|
func (api *StatusAPI) SelectAccount(address, password string) error {
|
2017-10-17 04:07:42 +07:00
|
|
|
// FIXME(oleg-raev): This method doesn't make stop, it rather resets its cells to an initial state
|
|
|
|
// and should be properly renamed, for example: ResetCells
|
|
|
|
api.b.jailManager.Stop()
|
2018-03-22 13:31:12 +01:00
|
|
|
return api.b.SelectAccount(address, password)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Logout clears whisper identities
|
|
|
|
func (api *StatusAPI) Logout() error {
|
2017-10-17 04:07:42 +07:00
|
|
|
api.b.jailManager.Stop()
|
2018-03-22 13:31:12 +01:00
|
|
|
return api.b.Logout()
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
2017-09-04 14:56:58 +02:00
|
|
|
// SendTransaction creates a new transaction and waits until it's complete.
|
|
|
|
func (api *StatusAPI) SendTransaction(ctx context.Context, args common.SendTxArgs) (gethcommon.Hash, error) {
|
|
|
|
return api.b.SendTransaction(ctx, args)
|
|
|
|
}
|
|
|
|
|
2017-05-16 15:09:52 +03:00
|
|
|
// CompleteTransaction instructs backend to complete sending of a given transaction
|
2017-09-04 14:56:58 +02:00
|
|
|
func (api *StatusAPI) CompleteTransaction(id common.QueuedTxID, password string) (gethcommon.Hash, error) {
|
|
|
|
return api.b.txQueueManager.CompleteTransaction(id, password)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// CompleteTransactions instructs backend to complete sending of multiple transactions
|
2018-01-05 22:58:17 +02:00
|
|
|
func (api *StatusAPI) CompleteTransactions(ids []common.QueuedTxID, password string) map[common.QueuedTxID]common.TransactionResult {
|
2017-09-04 14:56:58 +02:00
|
|
|
return api.b.txQueueManager.CompleteTransactions(ids, password)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// DiscardTransaction discards a given transaction from transaction queue
|
2017-09-04 14:56:58 +02:00
|
|
|
func (api *StatusAPI) DiscardTransaction(id common.QueuedTxID) error {
|
|
|
|
return api.b.txQueueManager.DiscardTransaction(id)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// DiscardTransactions discards given multiple transactions from transaction queue
|
2017-09-04 14:56:58 +02:00
|
|
|
func (api *StatusAPI) DiscardTransactions(ids []common.QueuedTxID) map[common.QueuedTxID]common.RawDiscardTransactionResult {
|
|
|
|
return api.b.txQueueManager.DiscardTransactions(ids)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
2017-11-23 13:37:59 +01:00
|
|
|
// JailParse creates a new jail cell context, with the given chatID as identifier.
|
|
|
|
// New context executes provided JavaScript code, right after the initialization.
|
|
|
|
// DEPRECATED in favour of CreateAndInitCell.
|
|
|
|
func (api *StatusAPI) JailParse(chatID string, js string) string {
|
|
|
|
return api.b.jailManager.Parse(chatID, js)
|
|
|
|
}
|
|
|
|
|
2017-11-07 18:36:42 +01:00
|
|
|
// CreateAndInitCell creates a new jail cell context, with the given chatID as identifier.
|
2017-05-16 15:09:52 +03:00
|
|
|
// New context executes provided JavaScript code, right after the initialization.
|
2017-11-07 18:36:42 +01:00
|
|
|
func (api *StatusAPI) CreateAndInitCell(chatID, js string) string {
|
|
|
|
return api.b.jailManager.CreateAndInitCell(chatID, js)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
2017-05-28 16:57:30 +03:00
|
|
|
// JailCall executes given JavaScript function w/i a jail cell context identified by the chatID.
|
2017-09-02 20:04:23 +03:00
|
|
|
func (api *StatusAPI) JailCall(chatID, this, args string) string {
|
|
|
|
return api.b.jailManager.Call(chatID, this, args)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
|
|
|
|
2017-11-07 18:36:42 +01:00
|
|
|
// JailExecute allows to run arbitrary JS code within a jail cell.
|
|
|
|
func (api *StatusAPI) JailExecute(chatID, code string) string {
|
|
|
|
return api.b.jailManager.Execute(chatID, code)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetJailBaseJS allows to setup initial JavaScript to be loaded on each jail.CreateAndInitCell().
|
|
|
|
func (api *StatusAPI) SetJailBaseJS(js string) {
|
|
|
|
api.b.jailManager.SetBaseJS(js)
|
2017-05-16 15:09:52 +03:00
|
|
|
}
|
2017-09-15 17:57:34 +02:00
|
|
|
|
2017-10-23 18:46:51 +03:00
|
|
|
// Notify sends a push notification to the device with the given token.
|
|
|
|
// @deprecated
|
|
|
|
func (api *StatusAPI) Notify(token string) string {
|
2018-03-20 14:35:28 -04:00
|
|
|
api.log.Debug("Notify", "token", token)
|
2017-10-23 18:46:51 +03:00
|
|
|
message := "Hello World1"
|
2017-10-11 16:51:43 +03:00
|
|
|
|
2017-10-23 18:46:51 +03:00
|
|
|
tokens := []string{token}
|
|
|
|
|
|
|
|
err := api.b.newNotification().Send(message, fcm.NotificationPayload{}, tokens...)
|
|
|
|
if err != nil {
|
2018-03-20 14:35:28 -04:00
|
|
|
api.log.Error("Notify failed", "error", err)
|
2017-10-10 18:30:56 +03:00
|
|
|
}
|
|
|
|
|
2017-10-23 18:46:51 +03:00
|
|
|
return token
|
|
|
|
}
|
|
|
|
|
|
|
|
// NotifyUsers send notifications to users.
|
|
|
|
func (api *StatusAPI) NotifyUsers(message string, payload fcm.NotificationPayload, tokens ...string) error {
|
2018-03-20 14:35:28 -04:00
|
|
|
api.log.Debug("Notify", "tokens", tokens)
|
2017-10-23 18:46:51 +03:00
|
|
|
|
|
|
|
err := api.b.newNotification().Send(message, payload, tokens...)
|
2017-10-11 16:51:43 +03:00
|
|
|
if err != nil {
|
2018-03-20 14:35:28 -04:00
|
|
|
api.log.Error("Notify failed", "error", err)
|
2017-10-11 16:51:43 +03:00
|
|
|
}
|
2017-09-15 17:57:34 +02:00
|
|
|
|
2017-10-12 17:31:39 +03:00
|
|
|
return err
|
2017-09-15 17:57:34 +02:00
|
|
|
}
|
2018-02-09 22:28:16 +02:00
|
|
|
|
|
|
|
// ConnectionChange handles network state changes logic.
|
|
|
|
func (api *StatusAPI) ConnectionChange(typ string, expensive bool) {
|
|
|
|
state := ConnectionState{
|
|
|
|
Type: NewConnectionType(typ),
|
|
|
|
Expensive: expensive,
|
|
|
|
}
|
|
|
|
if typ == "none" {
|
|
|
|
state.Offline = true
|
|
|
|
}
|
|
|
|
api.b.ConnectionChange(state)
|
|
|
|
}
|
2018-03-14 16:46:21 +01:00
|
|
|
|
|
|
|
// AppStateChange handles app state changes (background/foreground).
|
|
|
|
// state values: see https://facebook.github.io/react-native/docs/appstate.html
|
|
|
|
func (api *StatusAPI) AppStateChange(state string) {
|
|
|
|
appState, err := ParseAppState(state)
|
|
|
|
if err != nil {
|
2018-03-20 14:35:28 -04:00
|
|
|
log.Error("AppStateChange failed, ignoring", "error", err)
|
2018-03-14 16:46:21 +01:00
|
|
|
return // and do nothing
|
|
|
|
}
|
|
|
|
api.b.AppStateChange(appState)
|
|
|
|
}
|