mirror of https://github.com/embarklabs/embark.git
feat(@embark/whisper): Add Whisper client config
Add option in communication config to choose which Whisper client to use. Because Parity’s implementation of Whisper is not compatible with Whisper v6, and therefore web3.js in its current form, the following changes have been made: 1. remove any functionality associated with launching a Parity Whisper process. 2. Warn the user when the Parity Whisper client has been opted for in the communication config. 3. Return an error for API calls when Parity Whisper client has been opted for in the communication config. 4. Update Cockpit’s Communication module to show errors returned from API calls. 5. Update the messaging configuration documentation for the new communication client option.
This commit is contained in:
parent
446197baff
commit
bd4b110a78
|
@ -4,6 +4,7 @@ module.exports = {
|
||||||
enabled: true,
|
enabled: true,
|
||||||
provider: "whisper", // Communication provider. Currently, Embark only supports whisper
|
provider: "whisper", // Communication provider. Currently, Embark only supports whisper
|
||||||
available_providers: ["whisper"], // Array of available providers
|
available_providers: ["whisper"], // Array of available providers
|
||||||
|
client: "geth"
|
||||||
},
|
},
|
||||||
|
|
||||||
// default environment, merges with the settings in default
|
// default environment, merges with the settings in default
|
||||||
|
@ -35,12 +36,12 @@ module.exports = {
|
||||||
// "embark run custom_name"
|
// "embark run custom_name"
|
||||||
//custom_name: {
|
//custom_name: {
|
||||||
//}
|
//}
|
||||||
// Use this section when you need a specific symmetric or private keys in whisper
|
// Use this section when you need a specific symmetric or private keys in whisper
|
||||||
/*
|
/*
|
||||||
,keys: {
|
,keys: {
|
||||||
symmetricKey: "your_symmetric_key",// Symmetric key for message decryption
|
symmetricKey: "your_symmetric_key",// Symmetric key for message decryption
|
||||||
privateKey: "your_private_key" // Private Key to be used as a signing key and for message decryption
|
privateKey: "your_private_key" // Private Key to be used as a signing key and for message decryption
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
//}
|
//}
|
||||||
};
|
};
|
||||||
|
|
|
@ -4,6 +4,7 @@ module.exports = {
|
||||||
enabled: true,
|
enabled: true,
|
||||||
provider: "whisper", // Communication provider. Currently, Embark only supports whisper
|
provider: "whisper", // Communication provider. Currently, Embark only supports whisper
|
||||||
available_providers: ["whisper"], // Array of available providers
|
available_providers: ["whisper"], // Array of available providers
|
||||||
|
client: "geth"
|
||||||
},
|
},
|
||||||
|
|
||||||
// default environment, merges with the settings in default
|
// default environment, merges with the settings in default
|
||||||
|
@ -36,11 +37,11 @@ module.exports = {
|
||||||
// "embark run custom_name"
|
// "embark run custom_name"
|
||||||
//custom_name: {
|
//custom_name: {
|
||||||
//}
|
//}
|
||||||
// Use this section when you need a specific symmetric or private keys in whisper
|
// Use this section when you need a specific symmetric or private keys in whisper
|
||||||
/*
|
/*
|
||||||
,keys: {
|
,keys: {
|
||||||
symmetricKey: "your_symmetric_key",// Symmetric key for message decryption
|
symmetricKey: "your_symmetric_key",// Symmetric key for message decryption
|
||||||
privateKey: "your_private_key" // Private Key to be used as a signing key and for message decryption
|
privateKey: "your_private_key" // Private Key to be used as a signing key and for message decryption
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import PropTypes from "prop-types";
|
import PropTypes from "prop-types";
|
||||||
import React, {Component} from 'react';
|
import React, { Component } from 'react';
|
||||||
import {
|
import {
|
||||||
Button,
|
Button,
|
||||||
Card,
|
Card,
|
||||||
|
@ -13,6 +13,7 @@ import {
|
||||||
ListGroup,
|
ListGroup,
|
||||||
ListGroupItem
|
ListGroupItem
|
||||||
} from 'reactstrap';
|
} from 'reactstrap';
|
||||||
|
import Error from "./Error";
|
||||||
|
|
||||||
class Communication extends Component {
|
class Communication extends Component {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
|
@ -61,6 +62,7 @@ class Communication extends Component {
|
||||||
return (
|
return (
|
||||||
<Row className="justify-content-md-center">
|
<Row className="justify-content-md-center">
|
||||||
<Col xs="12" sm="9" lg="9">
|
<Col xs="12" sm="9" lg="9">
|
||||||
|
{this.props.error && this.props.error.error && <Error error={this.props.error.error} />}
|
||||||
<Card>
|
<Card>
|
||||||
<CardHeader>
|
<CardHeader>
|
||||||
<strong>Listen to channel</strong>
|
<strong>Listen to channel</strong>
|
||||||
|
@ -147,7 +149,8 @@ Communication.propTypes = {
|
||||||
sendMessage: PropTypes.func,
|
sendMessage: PropTypes.func,
|
||||||
listenToMessages: PropTypes.func,
|
listenToMessages: PropTypes.func,
|
||||||
subscriptions: PropTypes.array,
|
subscriptions: PropTypes.array,
|
||||||
channels: PropTypes.object
|
channels: PropTypes.object,
|
||||||
|
error: PropTypes.object
|
||||||
};
|
};
|
||||||
|
|
||||||
export default Communication;
|
export default Communication;
|
||||||
|
|
|
@ -5,7 +5,7 @@ import { Alert } from 'reactstrap';
|
||||||
import {messageSend, messageListen, versions} from "../actions";
|
import {messageSend, messageListen, versions} from "../actions";
|
||||||
import Communication from "../components/Communication";
|
import Communication from "../components/Communication";
|
||||||
import PageHead from "../components/PageHead";
|
import PageHead from "../components/PageHead";
|
||||||
import {getMessages, getMessageChannels, isOldWeb3, isWeb3Enabled} from "../reducers/selectors";
|
import { getMessages, getMessageChannels, isOldWeb3, isWeb3Enabled, getMessagesError } from "../reducers/selectors";
|
||||||
|
|
||||||
class CommunicationContainer extends Component {
|
class CommunicationContainer extends Component {
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
|
@ -37,9 +37,11 @@ class CommunicationContainer extends Component {
|
||||||
<React.Fragment>
|
<React.Fragment>
|
||||||
<PageHead title="Communication" description="Interact with the decentralised communication protocols configured for Embark (ie Whisper)" />
|
<PageHead title="Communication" description="Interact with the decentralised communication protocols configured for Embark (ie Whisper)" />
|
||||||
<Communication listenToMessages={(channel) => this.listenToChannel(channel)}
|
<Communication listenToMessages={(channel) => this.listenToChannel(channel)}
|
||||||
sendMessage={(channel, message) => this.sendMessage(channel, message)}
|
sendMessage={(channel, message) => this.sendMessage(channel, message)}
|
||||||
channels={this.props.messages}
|
channels={this.props.messages}
|
||||||
subscriptions={this.props.messageChannels}/>
|
subscriptions={this.props.messageChannels}
|
||||||
|
error={this.props.error}
|
||||||
|
/>
|
||||||
</React.Fragment>
|
</React.Fragment>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -60,7 +62,8 @@ CommunicationContainer.propTypes = {
|
||||||
isWeb3Enabled: PropTypes.bool,
|
isWeb3Enabled: PropTypes.bool,
|
||||||
messages: PropTypes.object,
|
messages: PropTypes.object,
|
||||||
messageChannels: PropTypes.array,
|
messageChannels: PropTypes.array,
|
||||||
fetchVersions: PropTypes.func
|
fetchVersions: PropTypes.func,
|
||||||
|
error: PropTypes.object
|
||||||
};
|
};
|
||||||
|
|
||||||
function mapStateToProps(state) {
|
function mapStateToProps(state) {
|
||||||
|
@ -68,7 +71,8 @@ function mapStateToProps(state) {
|
||||||
messages: getMessages(state),
|
messages: getMessages(state),
|
||||||
messageChannels: getMessageChannels(state),
|
messageChannels: getMessageChannels(state),
|
||||||
isOldWeb3: isOldWeb3(state),
|
isOldWeb3: isOldWeb3(state),
|
||||||
isWeb3Enabled: isWeb3Enabled(state)
|
isWeb3Enabled: isWeb3Enabled(state),
|
||||||
|
error: getMessagesError(state)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -150,6 +150,12 @@ export function getMessages(state) {
|
||||||
return messages;
|
return messages;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function getMessagesError(state) {
|
||||||
|
return {
|
||||||
|
error: state.errorEntities.messages
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
export function getFiddleDeploy(state) {
|
export function getFiddleDeploy(state) {
|
||||||
return {
|
return {
|
||||||
data: last(state.entities.fiddleDeploys),
|
data: last(state.entities.fiddleDeploys),
|
||||||
|
|
|
@ -571,7 +571,10 @@ export function *listenToMessages(action) {
|
||||||
const channel = yield call(createChannel, socket);
|
const channel = yield call(createChannel, socket);
|
||||||
while (true) {
|
while (true) {
|
||||||
const message = yield take(channel);
|
const message = yield take(channel);
|
||||||
yield put(actions.messageListen.success([{channel: action.messageChannels[0], message: message.data, time: message.time}]));
|
if (message.error) {
|
||||||
|
return yield put(actions.messageListen.failure(message.error));
|
||||||
|
}
|
||||||
|
yield put(actions.messageListen.success([{ channel: action.messageChannels[0], message: message.data, time: message.time }]));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -567,6 +567,7 @@ export class Config {
|
||||||
enabled: true,
|
enabled: true,
|
||||||
provider: "whisper",
|
provider: "whisper",
|
||||||
available_providers: ["whisper"],
|
available_providers: ["whisper"],
|
||||||
|
client: "geth",
|
||||||
connection: {
|
connection: {
|
||||||
host: defaultHost,
|
host: defaultHost,
|
||||||
port: 8547,
|
port: 8547,
|
||||||
|
|
|
@ -18,7 +18,7 @@ class Whisper {
|
||||||
this.webSocketsChannels = {};
|
this.webSocketsChannels = {};
|
||||||
this.modulesPath = dappPath(embark.config.embarkConfig.generationDir, constants.dappArtifacts.symlinkDir);
|
this.modulesPath = dappPath(embark.config.embarkConfig.generationDir, constants.dappArtifacts.symlinkDir);
|
||||||
|
|
||||||
if (!this.communicationConfig.enabled || this.blockchainConfig.client !== constants.blockchain.clients.geth) {
|
if (!this.communicationConfig.enabled || this.communicationConfig.client !== constants.blockchain.clients.geth) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
import EmbarkJS from "embarkjs";
|
import EmbarkJS from "embarkjs";
|
||||||
import EmbarkJSWhisper from "embarkjs-whisper";
|
import EmbarkJSWhisper from "embarkjs-whisper";
|
||||||
|
|
||||||
class API {
|
export const PARITY_WHISPER_ERROR = "Parity's implementation of Whisper is not compatible with Whisper v6 (and therefore web3.js). Try changing the communication config to use '{client: \"geth\"}' instead.";
|
||||||
|
|
||||||
|
export class Api {
|
||||||
|
|
||||||
constructor(embark) {
|
constructor(embark) {
|
||||||
this.embark = embark;
|
this.embark = embark;
|
||||||
|
@ -25,13 +27,8 @@ class API {
|
||||||
this.embark.registerAPICall(
|
this.embark.registerAPICall(
|
||||||
"post",
|
"post",
|
||||||
"/embark-api/communication/sendMessage",
|
"/embark-api/communication/sendMessage",
|
||||||
(req, res) => {
|
(_req, res) => {
|
||||||
EmbarkJS.Messages.sendMessage({ topic: req.body.topic, data: req.body.message }, (err, result) => {
|
res.status(500).send({ error: PARITY_WHISPER_ERROR });
|
||||||
if (err) {
|
|
||||||
return res.status(500).send({ error: err });
|
|
||||||
}
|
|
||||||
res.send(result);
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,13 +36,8 @@ class API {
|
||||||
this.embark.registerAPICall(
|
this.embark.registerAPICall(
|
||||||
"ws",
|
"ws",
|
||||||
"/embark-api/communication/listenTo/:topic",
|
"/embark-api/communication/listenTo/:topic",
|
||||||
(ws, req) => {
|
(ws, _req) => {
|
||||||
EmbarkJS.Messages.listenTo({ topic: req.params.topic }).subscribe(data => {
|
ws.send(JSON.stringify({ error: PARITY_WHISPER_ERROR }));
|
||||||
ws.send(JSON.stringify(data));
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = API;
|
|
||||||
|
|
|
@ -1,65 +0,0 @@
|
||||||
const WebSocket = require("ws");
|
|
||||||
const http = require("http");
|
|
||||||
|
|
||||||
const LIVENESS_CHECK = JSON.stringify({
|
|
||||||
jsonrpc: '2.0',
|
|
||||||
method: 'web3_clientVersion',
|
|
||||||
params:[],
|
|
||||||
id:42
|
|
||||||
});
|
|
||||||
|
|
||||||
// eslint-disable-next-line complexity
|
|
||||||
const parseAndRespond = (data, cb) => {
|
|
||||||
let resp;
|
|
||||||
try {
|
|
||||||
resp = JSON.parse(data);
|
|
||||||
if (resp.error) {
|
|
||||||
return cb(resp.error);
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
return cb("Version data is not valid JSON");
|
|
||||||
}
|
|
||||||
if (!resp || !resp.result) {
|
|
||||||
return cb("No version returned");
|
|
||||||
}
|
|
||||||
const result = resp.result.replace("//", "/");
|
|
||||||
const [_, version, __] = result.split("/");
|
|
||||||
cb(null, version);
|
|
||||||
};
|
|
||||||
|
|
||||||
const rpc = (host, port, cb) => {
|
|
||||||
const options = {
|
|
||||||
hostname: host, // TODO(andremedeiros): get from config
|
|
||||||
port, // TODO(andremedeiros): get from config
|
|
||||||
method: "POST",
|
|
||||||
timeout: 1000,
|
|
||||||
headers: {
|
|
||||||
"Content-Type": "application/json",
|
|
||||||
"Content-Length": Buffer.byteLength(LIVENESS_CHECK)
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const req = http.request(options, (res) => {
|
|
||||||
let data = "";
|
|
||||||
res.on("data", (chunk) => { data += chunk; });
|
|
||||||
res.on("end", () => parseAndRespond(data, cb));
|
|
||||||
});
|
|
||||||
req.on("error", (e) => cb(e));
|
|
||||||
req.write(LIVENESS_CHECK);
|
|
||||||
req.end();
|
|
||||||
};
|
|
||||||
|
|
||||||
const ws = (host, port, cb) => {
|
|
||||||
const conn = new WebSocket("ws://" + host + ":" + port);
|
|
||||||
conn.on("message", (data) => {
|
|
||||||
parseAndRespond(data, cb);
|
|
||||||
conn.close();
|
|
||||||
});
|
|
||||||
conn.on("open", () => conn.send(LIVENESS_CHECK));
|
|
||||||
conn.on("error", (e) => cb(e));
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
ws,
|
|
||||||
rpc
|
|
||||||
};
|
|
|
@ -1,8 +1,7 @@
|
||||||
import { __ } from "embark-i18n";
|
import { __ } from "embark-i18n";
|
||||||
import { dappPath, canonicalHost, defaultHost } from "embark-utils";
|
import { dappPath, canonicalHost, defaultHost } from "embark-utils";
|
||||||
const constants = require("embark-core/constants");
|
const constants = require("embark-core/constants");
|
||||||
const API = require("./api.js");
|
const { Api, PARITY_WHISPER_ERROR } = require("./api.js");
|
||||||
import { ws, rpc } from "./check.js";
|
|
||||||
|
|
||||||
class Whisper {
|
class Whisper {
|
||||||
constructor(embark, _options) {
|
constructor(embark, _options) {
|
||||||
|
@ -16,71 +15,33 @@ class Whisper {
|
||||||
this.webSocketsChannels = {};
|
this.webSocketsChannels = {};
|
||||||
this.modulesPath = dappPath(embark.config.embarkConfig.generationDir, constants.dappArtifacts.symlinkDir);
|
this.modulesPath = dappPath(embark.config.embarkConfig.generationDir, constants.dappArtifacts.symlinkDir);
|
||||||
|
|
||||||
if (!this.communicationConfig.enabled || this.blockchainConfig.client !== constants.blockchain.clients.parity) {
|
if (!this.communicationConfig.enabled || this.communicationConfig.client !== constants.blockchain.clients.parity) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.api = new API(embark);
|
|
||||||
this.whisperNodes = {};
|
this.whisperNodes = {};
|
||||||
|
|
||||||
this.events.request("embarkjs:plugin:register", "messages", "whisper", "embarkjs-whisper");
|
this.events.request("embarkjs:plugin:register", "messages", "whisper", "embarkjs-whisper");
|
||||||
this.events.request("embarkjs:console:register", "messages", "whisper", "embarkjs-whisper");
|
this.events.request("embarkjs:console:register", "messages", "whisper", "embarkjs-whisper");
|
||||||
|
|
||||||
this.events.request("communication:node:register", "whisper", (readyCb) => {
|
this.events.request("communication:node:register", "whisper", (readyCb) => {
|
||||||
this.events.request("processes:register", "communication", {
|
this.logger.warn(PARITY_WHISPER_ERROR);
|
||||||
launchFn: (cb) => {
|
readyCb();
|
||||||
this.startWhisperNode(cb);
|
|
||||||
},
|
|
||||||
stopFn: (cb) => {
|
|
||||||
this.stopWhisperNode(cb);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
this.events.request("processes:launch", "communication", (err) => {
|
|
||||||
if (err) {
|
|
||||||
this.logger.error(`Error launching whisper process: ${err.message || err}`);
|
|
||||||
}
|
|
||||||
readyCb();
|
|
||||||
});
|
|
||||||
|
|
||||||
this.registerServiceCheck();
|
this.registerServiceCheck();
|
||||||
});
|
});
|
||||||
|
|
||||||
this.events.on("communication:started", () => {
|
this.events.on("communication:started", () => {
|
||||||
this.api = new API(embark);
|
this.api = new Api(embark);
|
||||||
this.api.registerAPICalls();
|
this.api.registerAPICalls();
|
||||||
this.connectEmbarkJSProvider();
|
this.connectEmbarkJSProvider();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
_getNodeState(err, version, cb) {
|
|
||||||
if (err) return cb({ name: "Whisper node not found", status: "off" });
|
|
||||||
|
|
||||||
let nodeName = "Parity";
|
|
||||||
let versionNumber = version.split("-")[0];
|
|
||||||
let name = nodeName + " " + versionNumber + " (Whisper)";
|
|
||||||
return cb({ name, status: "on" });
|
|
||||||
}
|
|
||||||
|
|
||||||
registerServiceCheck() {
|
registerServiceCheck() {
|
||||||
this.events.request("services:register", "Whisper", (cb) => {
|
this.events.request("services:register", "Whisper", (cb) => {
|
||||||
const { host, port, type } = this.communicationConfig.connection;
|
cb({ name: "Whisper Parity", status: "off" });
|
||||||
if (type === "ws") {
|
}, 1000 * 60 * 30, "off");
|
||||||
return ws(host, port, (err, version) => this._getNodeState(err, version, cb));
|
|
||||||
}
|
|
||||||
rpc(host, port, (err, version) => this._getNodeState(err, version, cb));
|
|
||||||
|
|
||||||
}, 5000, "off");
|
|
||||||
}
|
|
||||||
|
|
||||||
startWhisperNode(callback) {
|
|
||||||
this.logger.info(`Whisper node has already been started with the Parity blockchain.`);
|
|
||||||
callback();
|
|
||||||
}
|
|
||||||
|
|
||||||
stopWhisperNode(cb) {
|
|
||||||
this.logger.warn(`Cannot stop Whisper process as it has been started with the Parity blockchain.`);
|
|
||||||
cb();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// esline-disable-next-line complexity
|
// esline-disable-next-line complexity
|
||||||
|
|
|
@ -28,4 +28,5 @@ Option | Type: `default` | Value
|
||||||
`enabled` | boolean: `true/false` | To enable or completely disable communication support
|
`enabled` | boolean: `true/false` | To enable or completely disable communication support
|
||||||
`provider` | string: `whisper` | Desired provider to automatically connect to in the dapp.
|
`provider` | string: `whisper` | Desired provider to automatically connect to in the dapp.
|
||||||
`available_providers` | array: `["whisper"]` | List of communication platforms to be supported in the dapp. This will affect what's available with the EmbarkJS library in the dapp so if you don't need Whisper for example, removing it from this will considerably reduce the file size of the generated JS code.
|
`available_providers` | array: `["whisper"]` | List of communication platforms to be supported in the dapp. This will affect what's available with the EmbarkJS library in the dapp so if you don't need Whisper for example, removing it from this will considerably reduce the file size of the generated JS code.
|
||||||
|
`client` | string: `geth/parity` | Desired Whisper client for Embark to start. **NOTE:** Parity's implementation of Whisper does not currently adhere to Whisper v6 standards and thus is not supported by `web3.js`.
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue