You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

477 lines
12 KiB

6 years ago
/*!
* accounts/notification-web-sockets.js
* Copyright © 2019 Katana Cryptographic Ltd. All Rights Reserved.
*/
'use strict'
const _ = require('lodash')
const LRU = require('lru-cache')
const WebSocket = require('websocket')
const Logger = require('../lib/logger')
const network = require('../lib/bitcoin/network')
const keys = require('../keys')[network.key]
const apiHelper = require('./api-helper')
const status = require('./status')
const authMgr = require('../lib/auth/authorizations-manager')
const debug = process.argv.indexOf('ws-debug') > -1
6 years ago
/**
* A class providing a notifications server over web sockets
*/
class NotificationsService {
/**
* Constructor
* @param {object} server - listening instance of a http server
*/
constructor(server) {
// Web sockets server
this.ws = null
// Dictionary of connections
this.conn = {}
// Dictionary of subscriptions
this.subs = {}
// Dictionary mapping addresses to pubkeys
this.cachePubKeys = {}
// Cache registering the most recent subscriptions received
// Used to filter multiple subscriptions sent by external apps.
this.cacheSubs = new LRU({
6 years ago
// Maximum number of subscriptions to store in cache
// Estimate: 1000 clients with an average of 5 subscriptions
max: 5000,
// Function used to compute length of item
length: (n, key) => 1,
// Maximum age for items in the cache (1mn)
maxAge: 60000
})
// Initialize the web socket server
this._initWSServer(server)
}
/**
* Initialize the web sockets server
* @param {object} server - listening instance of a http server
*/
_initWSServer(server) {
this.ws = new WebSocket.server({httpServer: server})
Logger.info('API : Created WebSocket server')
6 years ago
this.ws.on('request', req => {
try {
let conn = req.accept(null, req.origin)
conn.id = status.sessions++
conn.subs = []
debug && Logger.info(`API : Client ${conn.id} connected`)
6 years ago
conn.on('close', () => {
this._closeWSConnection(conn, false)
})
conn.on('error', err => {
Logger.error(err, `API : NotificationsService : Error on connection ${conn.id}`)
6 years ago
if (conn.connected)
this._closeWSConnection(conn, true)
})
conn.on('message', msg => {
if (msg.type === 'utf8')
6 years ago
this._handleWSMessage(msg.utf8Data, conn)
else
this._closeWSConnection(conn, true)
})
this.conn[conn.id] = conn
status.clients = status.clients + 1
status.maxConn = Math.max(status.maxConn, Object.keys(this.conn).length)
} catch(e) {
Logger.error(e, `API : NotificationsService._initWSServer() : Error during request accept`)
6 years ago
}
})
}
6 years ago
/**
* Close a web sockets connection
* @param {object} conn - web socket connection
* @param {boolean} forcedClose - true if close initiated by server
*/
_closeWSConnection(conn, forcedClose) {
try {
for (let topic of conn.subs) {
this._unsub(topic, conn.id)
// Close initiated by client, remove subscriptions from cache
if (!forcedClose && this.cacheSubs.has(topic))
this.cacheSubs.del(topic)
}
if (this.conn[conn.id]) {
delete this.conn[conn.id]
status.clients = status.clients - 1
}
// Close initiated by server, drop the connection
if (forcedClose && conn.connected)
conn.drop(1008, 'Get out of here!')
debug && Logger.info(`API : Client ${conn.id} disconnected`)
6 years ago
} catch(e) {
Logger.error(e, 'API : NotificationsService._closeWSConnection()')
6 years ago
}
}
/**
* Filter potential duplicate subscriptions
* @param {string} msg - subscription received
* @returns {boolean} returns false if it's a duplicate, true otherwise.
*/
_filterWSMessage(msg) {
if (this.cacheSubs.has(msg)) {
debug && Logger.info('API : Duplicate subscriptions detected')
6 years ago
return false
} else {
this.cacheSubs.set(msg, true)
return true
}
}
/**
* Handle messages received over the web sockets
* (subscriptions)
* @param {string} msg - subscription received
* @param {object} conn - connection
*/
_handleWSMessage(msg, conn) {
try {
debug && Logger.info(`API : Received from client ${conn.id}: ${msg}`)
6 years ago
const data = JSON.parse(msg)
// Check authentication (if needed)
if (authMgr.authActive && authMgr.isMandatory) {
try {
authMgr.isAuthenticated(data.at)
6 years ago
} catch(e) {
this.notifyAuthError(e, conn.id)
return
}
6 years ago
}
switch(data.op) {
case 'ping':
conn.sendUTF('{"op": "pong"}')
break
case 'addr_sub':
if (data.addr) {
// Check for potential flood by clients
// subscribing for the same xpub again and again
if (this._filterWSMessage(data.addr))
this._entitysub(data.addr, conn)
else
this._closeWSConnection(conn, true)
}
break
case 'blocks_sub':
this._addsub('block', conn)
break
}
} catch(e) {
Logger.error(e, 'API : NotificationsService._handleWSMessage() : WebSocket message error')
6 years ago
}
}
/**
* Subscribe to a list of addresses/xpubs/pubkeys
* @param {string} topic - topic
* @param {object} conn - connection asking for subscription
*/
_entitysub(topic, conn) {
const valid = apiHelper.parseEntities(topic)
for (let a in valid.addrs) {
const address = valid.addrs[a]
this._addsub(address, conn)
if (valid.pubkeys[a]) {
this.cachePubKeys[address] = valid.pubkeys[a]
}
}
for (let xpub of valid.xpubs)
this._addsub(xpub, conn)
}
/**
* Subscribe to a topic
* @param {string} topic - topic
* @param {object} conn - connection asking for subscription
*/
_addsub(topic, conn) {
if (conn.subs.indexOf(topic) >= 0)
return false
conn.subs.push(topic)
if (!this.subs[topic])
this.subs[topic] = []
this.subs[topic].push(conn.id)
debug && Logger.info(`API : Client ${conn.id} subscribed to ${topic}`)
6 years ago
}
/**
* Unsubscribe from a topic
* @param {string} topic - topic
* @param {number} cid - client id
6 years ago
*/
_unsub(topic, cid) {
if (!this.subs[topic])
return false
const index = this.subs[topic].indexOf(cid)
if (index < 0)
6 years ago
return false
this.subs[topic].splice(index, 1)
if (this.subs[topic].length === 0) {
6 years ago
delete this.subs[topic]
if (this.cachePubKeys.hasOwnProperty(topic))
delete this.cachePubKeys[topic]
}
return true
}
/**
* Dispatch a notification to all clients
* who have subscribed to a topic
* @param {string} topic - topic
* @param {string} msg - content of the notification
*/
dispatch(topic, msg) {
if (!this.subs[topic])
return
for (let cid of this.subs[topic]) {
if (!this.conn[cid])
6 years ago
continue
try {
this.conn[cid].sendUTF(msg)
} catch(e) {
Logger.error(e, `API : NotificationsService.dispatch() : Error sending dispatch for ${topic} to client ${cid}`)
6 years ago
}
}
}
/**
* Dispatch notifications for a new block
* @param {string} header - block header
*/
notifyBlock(header) {
try {
const data = {
op: 'block',
x: header
}
this.dispatch('block', JSON.stringify(data))
} catch(e) {
Logger.error(e, `API : NotificationsService.notifyBlock()`)
6 years ago
}
}
/**
* Dispatch notifications for a transaction
*
* Transaction notification operates within these constraints:
* 1. Notify each client ONCE of a relevant transaction
* 2. Maintain privacy of other parties when transactions are between clients
*
* Clients subscribe to a list of xpubs and addresses. Transactions identify
* address and xpub if available on inputs and outputs, omitting inputs and
* outputs for untracked addresses.
*
* Example:
* tx
* inputs
* addr1
* xpub2
* outputs
* xpub1
* xpub2
* addr2
* xpub3
*
* subs
* addr1: client1, client2
* addr2: client1
* xpub1: client1
* xpub2: client2
* xpub4: client3
*
* client1: addr1, addr2, xpub1
* client2: addr1, xpub2
* client3: xpub4
*
* tx -> client1
* inputs
* addr1
* outputs
* xpub1
* addr2
*
* tx -> client2
* inputs
* addr1
* xpub2
* outputs
* xpub2
*
* @param {object} tx - transaction
*
* @note Synchronous processing done by this method
* may become a bottleneck in the future if under heavy load.
* Split in multiple async calls might make sense.
*/
notifyTransaction(tx) {
try {
// Topics extracted from the transaction
const topics = {}
// Client subscriptions: {[cid]: [topic1, topic2, ...]}
const clients = {}
// Extract topics from the inputs
for (let i in tx.inputs) {
let input = tx.inputs[i]
let topic = null
if (input.prev_out) {
// Topic is either xpub or addr. Should it be both?
if (input.prev_out.xpub) {
topic = input.prev_out.xpub.m
} else if (input.prev_out.addr) {
topic = input.prev_out.addr
}
}
if (this.subs[topic]) {
topics[topic] = true
// Add topic information to the input
input.topic = topic
}
}
// Extract topics from the outputs
for (let o in tx.out) {
let output = tx.out[o]
let topic = null
if (output.xpub) {
topic = output.xpub.m
} else if (output.addr) {
topic = output.addr
}
if (this.subs[topic]) {
topics[topic] = true
// Add topic information to the output
output.topic = topic
}
}
for (let topic in topics) {
for (let cid of this.subs[topic]) {
if (!clients[cid])
clients[cid] = []
if (clients[cid].indexOf(topic) === -1)
6 years ago
clients[cid].push(topic)
}
}
// Tailor a transaction for each client
for (let cid in clients) {
const ctx = _.cloneDeep(tx)
ctx.inputs = []
ctx.out = []
// List of topics relevant to this client
const clientTopics = clients[cid]
// Check for topic information on inputs & outputs (added above)
for (let input of tx.inputs) {
const topic = input.topic
if (topic && clientTopics.indexOf(topic) > -1) {
const cin = _.cloneDeep(input)
delete cin.topic
if (this.cachePubKeys.hasOwnProperty(topic))
cin.pubkey = this.cachePubKeys[topic]
ctx.inputs.push(cin)
}
}
for (let output of tx.out) {
const topic = output.topic
if (topic && clientTopics.indexOf(topic) > -1) {
const cout = _.cloneDeep(output)
delete cout.topic
if (this.cachePubKeys.hasOwnProperty(topic))
cout.pubkey = this.cachePubKeys[topic]
ctx.out.push(cout)
}
}
// Move on if the custom transaction has no inputs or outputs
if (ctx.inputs.length === 0 && ctx.out.length === 0)
6 years ago
continue
// Send custom transaction to client
const data = {
op: 'utx',
x: ctx
}
try {
this.conn[cid].sendUTF(JSON.stringify(data))
debug && Logger.error(`API : Sent ctx ${ctx.hash} to client ${cid}`)
6 years ago
} catch(e) {
Logger.error(e, `API : NotificationsService.notifyTransaction() : Trouble sending ctx to client ${cid}`)
6 years ago
}
}
} catch(e) {
Logger.error(e, `API : NotificationsService.notifyTransaction()`)
6 years ago
}
}
/**
* Dispatch notification for an authentication error
* @param {string} err - error
* @param {number} cid - connection id
6 years ago
*/
notifyAuthError(err, cid) {
const data = {
op: 'error',
msg: err
}
try {
this.conn[cid].sendUTF(JSON.stringify(data))
debug && Logger.error(`API : Sent authentication error to client ${cid}`)
6 years ago
} catch(e) {
Logger.error(e, `API : NotificationsService.notifyAuthError() : Trouble sending authentication error to client ${cid}`)
6 years ago
}
}
6 years ago
}
module.exports = NotificationsService