2013-11-08 12:41:04 -05:00
|
|
|
// Copyright (c) 2013 Conformal Systems LLC.
|
|
|
|
// Use of this source code is governed by an ISC
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package btcws
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"github.com/conformal/btcjson"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// ErrNtfnUnexpected describes an error where an unexpected
|
|
|
|
// notification is received when unmarshaling into a concrete
|
|
|
|
// Notification variable.
|
|
|
|
ErrNtfnUnexpected = errors.New("notification unexpected")
|
|
|
|
|
|
|
|
// ErrNtfnNotFound describes an error where a parser does not
|
|
|
|
// handle unmarshaling a notification.
|
|
|
|
ErrNtfnNotFound = errors.New("notification not found")
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
// BlockConnectedNtfnId is the id of the btcd blockconnected
|
|
|
|
// notification.
|
2013-11-08 12:42:32 -05:00
|
|
|
BlockConnectedNtfnId = "btcd:blockconnected"
|
2013-11-08 12:41:04 -05:00
|
|
|
|
|
|
|
// BlockDisconnectedNtfnId is the id of the btcd blockdisconnected
|
|
|
|
// notification.
|
|
|
|
BlockDisconnectedNtfnId = "btcd:blockdisconnected"
|
|
|
|
|
2013-12-02 17:32:17 -05:00
|
|
|
// TxMinedNtfnId is the id of the btcd txmined notification.
|
2013-11-08 12:42:32 -05:00
|
|
|
TxMinedNtfnId = "btcd:txmined"
|
2013-12-02 17:32:17 -05:00
|
|
|
|
|
|
|
// TxNtfnId is the id of the btcwallet newtx notification.
|
|
|
|
TxNtfnId = "btcwallet:newtx"
|
2013-11-08 12:41:04 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
type newNtfnFn func() Notification
|
|
|
|
|
|
|
|
func newBlockConnectedNtfn() Notification {
|
|
|
|
return &BlockConnectedNtfn{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func newBlockDisconnectedNtfn() Notification {
|
|
|
|
return &BlockDisconnectedNtfn{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func newTxMinedNtfn() Notification {
|
|
|
|
return &TxMinedNtfn{}
|
|
|
|
}
|
|
|
|
|
2013-12-02 17:32:17 -05:00
|
|
|
func newTxNtfn() Notification {
|
|
|
|
return &TxNtfn{}
|
|
|
|
}
|
|
|
|
|
2013-11-08 12:41:04 -05:00
|
|
|
var newNtfnFns = map[string]newNtfnFn{
|
|
|
|
BlockConnectedNtfnId: newBlockConnectedNtfn,
|
|
|
|
BlockDisconnectedNtfnId: newBlockDisconnectedNtfn,
|
|
|
|
TxMinedNtfnId: newTxMinedNtfn,
|
2013-12-02 17:32:17 -05:00
|
|
|
TxNtfnId: newTxNtfn,
|
2013-11-08 12:41:04 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// ParseMarshaledNtfn attempts to unmarshal a marshaled notification
|
|
|
|
// into the notification described by id.
|
|
|
|
func ParseMarshaledNtfn(id string, b []byte) (Notification, error) {
|
|
|
|
if newFn, ok := newNtfnFns[id]; ok {
|
|
|
|
n := newFn()
|
|
|
|
if err := n.UnmarshalJSON(b); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return n, nil
|
|
|
|
}
|
|
|
|
return nil, ErrNtfnNotFound
|
|
|
|
}
|
|
|
|
|
|
|
|
// Notification is an interface implemented by all notification types.
|
|
|
|
type Notification interface {
|
|
|
|
json.Marshaler
|
|
|
|
json.Unmarshaler
|
|
|
|
Id() interface{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// BlockConnectedNtfn is a type handling custom marshaling and
|
|
|
|
// unmarshaling of blockconnected JSON websocket notifications.
|
|
|
|
type BlockConnectedNtfn struct {
|
|
|
|
Hash string `json:"hash"`
|
|
|
|
Height int32 `json:"height"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type blockConnectedResult BlockConnectedNtfn
|
|
|
|
|
|
|
|
// Enforce that BlockConnectedNtfn satisfies the Notification interface.
|
|
|
|
var _ Notification = &BlockConnectedNtfn{}
|
|
|
|
|
|
|
|
// NewBlockConnectedNtfn creates a new BlockConnectedNtfn.
|
|
|
|
func NewBlockConnectedNtfn(hash string, height int32) *BlockConnectedNtfn {
|
|
|
|
return &BlockConnectedNtfn{
|
|
|
|
Hash: hash,
|
|
|
|
Height: height,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-08 14:13:47 -05:00
|
|
|
// Id satisifies the Notification interface by returning the id of the
|
2013-11-08 12:41:04 -05:00
|
|
|
// notification.
|
|
|
|
func (n *BlockConnectedNtfn) Id() interface{} {
|
|
|
|
return BlockConnectedNtfnId
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON returns the JSON encoding of n. Part of the Notification
|
|
|
|
// interface.
|
|
|
|
func (n *BlockConnectedNtfn) MarshalJSON() ([]byte, error) {
|
|
|
|
id := n.Id()
|
|
|
|
reply := btcjson.Reply{
|
|
|
|
Result: *n,
|
|
|
|
Id: &id,
|
|
|
|
}
|
|
|
|
return json.Marshal(reply)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalJSON unmarshals the JSON encoding of n into n. Part of
|
|
|
|
// the Notification interface.
|
|
|
|
func (n *BlockConnectedNtfn) UnmarshalJSON(b []byte) error {
|
|
|
|
var ntfn struct {
|
|
|
|
Result blockConnectedResult `json:"result"`
|
|
|
|
Error *btcjson.Error `json:"error"`
|
|
|
|
Id interface{} `json:"id"`
|
|
|
|
}
|
|
|
|
if err := json.Unmarshal(b, &ntfn); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Notification IDs must match expected.
|
|
|
|
if n.Id() != ntfn.Id {
|
|
|
|
return ErrNtfnUnexpected
|
|
|
|
}
|
|
|
|
|
|
|
|
*n = BlockConnectedNtfn(ntfn.Result)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// BlockDisconnectedNtfn is a type handling custom marshaling and
|
|
|
|
// unmarshaling of blockdisconnected JSON websocket notifications.
|
|
|
|
type BlockDisconnectedNtfn struct {
|
|
|
|
Hash string `json:"hash"`
|
|
|
|
Height int32 `json:"height"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type blockDisconnectedResult BlockDisconnectedNtfn
|
|
|
|
|
|
|
|
// Enforce that BlockDisconnectedNtfn satisfies the Notification interface.
|
|
|
|
var _ Notification = &BlockDisconnectedNtfn{}
|
|
|
|
|
|
|
|
// NewBlockDisconnectedNtfn creates a new BlockConnectedNtfn.
|
|
|
|
func NewBlockDisconnectedNtfn(hash string, height int32) *BlockDisconnectedNtfn {
|
|
|
|
return &BlockDisconnectedNtfn{
|
|
|
|
Hash: hash,
|
|
|
|
Height: height,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Id satisifies the Notification interface by returning the id of the
|
|
|
|
// notification.
|
|
|
|
func (n *BlockDisconnectedNtfn) Id() interface{} {
|
|
|
|
return BlockDisconnectedNtfnId
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON returns the JSON encoding of n. Part of the Notification
|
|
|
|
// interface.
|
|
|
|
func (n *BlockDisconnectedNtfn) MarshalJSON() ([]byte, error) {
|
|
|
|
id := n.Id()
|
|
|
|
reply := btcjson.Reply{
|
|
|
|
Result: *n,
|
|
|
|
Id: &id,
|
|
|
|
}
|
|
|
|
return json.Marshal(reply)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalJSON unmarshals the JSON encoding of n into n. Part of
|
|
|
|
// the Notification interface.
|
|
|
|
func (n *BlockDisconnectedNtfn) UnmarshalJSON(b []byte) error {
|
|
|
|
var ntfn struct {
|
|
|
|
Result blockDisconnectedResult `json:"result"`
|
|
|
|
Error *btcjson.Error `json:"error"`
|
|
|
|
Id interface{} `json:"id"`
|
|
|
|
}
|
|
|
|
if err := json.Unmarshal(b, &ntfn); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Notification IDs must match expected.
|
|
|
|
if n.Id() != ntfn.Id {
|
|
|
|
return ErrNtfnUnexpected
|
|
|
|
}
|
|
|
|
|
|
|
|
*n = BlockDisconnectedNtfn(ntfn.Result)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// TxMinedNtfn is a type handling custom marshaling and
|
|
|
|
// unmarshaling of txmined JSON websocket notifications.
|
|
|
|
type TxMinedNtfn struct {
|
2013-11-25 12:54:01 -05:00
|
|
|
TxID string `json:"txid"`
|
|
|
|
BlockHash string `json:"blockhash"`
|
|
|
|
BlockHeight int32 `json:"blockheight"`
|
|
|
|
BlockTime int64 `json:"blocktime"`
|
|
|
|
Index int `json:"index"`
|
2013-11-08 12:41:04 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
type txMinedResult TxMinedNtfn
|
|
|
|
|
|
|
|
// Enforce that TxMinedNtfn satisfies the Notification interface.
|
|
|
|
var _ Notification = &TxMinedNtfn{}
|
|
|
|
|
|
|
|
// NewTxMinedNtfn creates a new TxMinedNtfn.
|
2013-11-25 12:54:01 -05:00
|
|
|
func NewTxMinedNtfn(txid, blockhash string, blockheight int32,
|
|
|
|
blocktime int64, index int) *TxMinedNtfn {
|
|
|
|
|
|
|
|
return &TxMinedNtfn{
|
|
|
|
TxID: txid,
|
|
|
|
BlockHash: blockhash,
|
|
|
|
BlockHeight: blockheight,
|
|
|
|
BlockTime: blocktime,
|
|
|
|
Index: index,
|
|
|
|
}
|
2013-11-08 12:41:04 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Id satisifies the Notification interface by returning the id of the
|
|
|
|
// notification.
|
|
|
|
func (n *TxMinedNtfn) Id() interface{} {
|
|
|
|
return TxMinedNtfnId
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON returns the JSON encoding of n. Part of the Notification
|
|
|
|
// interface.
|
|
|
|
func (n *TxMinedNtfn) MarshalJSON() ([]byte, error) {
|
|
|
|
id := n.Id()
|
|
|
|
reply := btcjson.Reply{
|
2013-11-08 13:03:34 -05:00
|
|
|
Result: *n,
|
2013-11-08 12:41:04 -05:00
|
|
|
Id: &id,
|
|
|
|
}
|
|
|
|
return json.Marshal(reply)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalJSON unmarshals the JSON encoding of n into n. Part of
|
|
|
|
// the Notification interface.
|
|
|
|
func (n *TxMinedNtfn) UnmarshalJSON(b []byte) error {
|
|
|
|
var ntfn struct {
|
|
|
|
Result txMinedResult `json:"result"`
|
|
|
|
Error *btcjson.Error `json:"error"`
|
|
|
|
Id interface{} `json:"id"`
|
|
|
|
}
|
|
|
|
if err := json.Unmarshal(b, &ntfn); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Notification IDs must match expected.
|
|
|
|
if n.Id() != ntfn.Id {
|
|
|
|
return ErrNtfnUnexpected
|
|
|
|
}
|
|
|
|
|
|
|
|
*n = TxMinedNtfn(ntfn.Result)
|
|
|
|
return nil
|
|
|
|
}
|
2013-12-02 17:32:17 -05:00
|
|
|
|
|
|
|
// TxNtfn is a type handling custom marshaling and
|
|
|
|
// unmarshaling of newtx JSON websocket notifications.
|
|
|
|
type TxNtfn struct {
|
2013-12-02 17:32:36 -05:00
|
|
|
Account string `json:"account"`
|
2013-12-02 17:32:17 -05:00
|
|
|
Details map[string]interface{} `json:"details"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type txNtfnResult TxNtfn
|
|
|
|
|
|
|
|
// Enforce that TxNtfn satisfies the Notification interface.
|
|
|
|
var _ Notification = &TxNtfn{}
|
|
|
|
|
|
|
|
// NewTxNtfn creates a new TxNtfn.
|
|
|
|
func NewTxNtfn(account string, details map[string]interface{}) *TxNtfn {
|
|
|
|
return &TxNtfn{
|
|
|
|
Account: account,
|
|
|
|
Details: details,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Id satisifies the Notification interface by returning the id of the
|
|
|
|
// notification.
|
|
|
|
func (n *TxNtfn) Id() interface{} {
|
|
|
|
return TxNtfnId
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalJSON returns the JSON encoding of n. Part of the Notification
|
|
|
|
// interface.
|
|
|
|
func (n *TxNtfn) MarshalJSON() ([]byte, error) {
|
|
|
|
id := n.Id()
|
|
|
|
reply := btcjson.Reply{
|
|
|
|
Result: *n,
|
|
|
|
Id: &id,
|
|
|
|
}
|
|
|
|
return json.Marshal(reply)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UnmarshalJSON unmarshals the JSON encoding of n into n. Part of
|
|
|
|
// the Notification interface.
|
|
|
|
func (n *TxNtfn) UnmarshalJSON(b []byte) error {
|
|
|
|
var ntfn struct {
|
|
|
|
Result txNtfnResult `json:"result"`
|
|
|
|
Error *btcjson.Error `json:"error"`
|
|
|
|
Id interface{} `json:"id"`
|
|
|
|
}
|
|
|
|
if err := json.Unmarshal(b, &ntfn); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Notification IDs must match expected.
|
|
|
|
if n.Id() != ntfn.Id {
|
|
|
|
return ErrNtfnUnexpected
|
|
|
|
}
|
|
|
|
|
|
|
|
*n = TxNtfn(ntfn.Result)
|
|
|
|
return nil
|
|
|
|
}
|