118 lines
4.8 KiB
JavaScript
Raw Normal View History

2017-12-28 12:40:11 -05:00
let utils = require('../../utils/utils.js');
let fs = require('../../core/fs.js');
let Web3 = require('web3');
2017-12-28 12:40:11 -05:00
2018-07-15 19:36:52 -05:00
const {canonicalHost, defaultHost} = require('../../utils/host');
2017-12-28 12:40:11 -05:00
class Whisper {
2018-06-01 19:35:41 -04:00
constructor(embark, _options) {
2017-12-28 12:40:11 -05:00
this.logger = embark.logger;
this.events = embark.events;
2018-05-30 12:26:49 -04:00
this.communicationConfig = embark.config.communicationConfig;
this.web3 = new Web3();
2017-12-28 12:40:11 -05:00
this.embark = embark;
if (!this.communicationConfig.enabled) {
return;
}
this.connectToProvider();
2017-12-28 12:40:11 -05:00
this.setServiceCheck();
this.addWhisperToEmbarkJS();
2017-12-28 18:10:43 -05:00
this.addSetProvider();
2017-12-28 12:40:11 -05:00
}
connectToProvider() {
2018-06-01 19:12:54 -04:00
let {host, port} = this.communicationConfig.connection;
let web3Endpoint = 'ws://' + host + ':' + port;
// Note: dont't pass to the provider things like {headers: {Origin: "embark"}}. Origin header is for browser to fill
// to protect user, it has no meaning if it is used server-side. See here for more details: https://github.com/ethereum/go-ethereum/issues/16608
// Moreover, Parity reject origins that are not urls so if you try to connect with Origin: "embark" it gives the followin error:
// << Blocked connection to WebSockets server from untrusted origin: Some("embark") >>
// The best choice is to use void origin, BUT Geth rejects void origin, so to keep both clients happy we can use http://embark
this.web3.setProvider(new Web3.providers.WebsocketProvider(web3Endpoint, {headers: {Origin: "http://embark"}}));
}
2017-12-28 12:40:11 -05:00
setServiceCheck() {
const self = this;
self.events.request("services:register", 'Whisper', function(cb) {
if (!self.web3.currentProvider || self.web3.currentProvider.connection.readyState !== 1) {
return self.connectToProvider();
}
// 1) Parity does not implement shh_version JSON-RPC method
// 2) web3 1.0 still does not implement web3_clientVersion
// so we must do all by our own
self.web3._requestManager.send({method: 'web3_clientVersion', params: []}, (err, clientVersion) => {
if (err) return cb(err);
if (clientVersion.indexOf("Parity-Ethereum//v2") === 0) {
// This is Parity
return self.web3.shh.getInfo(function(err) {
if (err) {
return cb({name: 'Whisper', status: 'off'});
}
// TOFIX Assume Whisper v6 until there's a way to understand it via JSON-RPC
return cb({name: 'Whisper (version 6)', status: 'on'});
});
}
// Assume it is a Geth compliant client
self.web3.shh.getVersion(function(err, version) {
if (err || version == "2") {
return cb({name: 'Whisper', status: 'off'});
}
return cb({name: 'Whisper (version ' + version + ')', status: 'on'});
});
2017-12-28 12:40:11 -05:00
});
});
}
addWhisperToEmbarkJS() {
2017-12-30 20:42:52 -05:00
const self = this;
2017-12-28 18:10:43 -05:00
// TODO: make this a shouldAdd condition
2017-12-28 12:40:11 -05:00
if (this.communicationConfig === {}) {
return;
}
if ((this.communicationConfig.available_providers.indexOf('whisper') < 0) && (this.communicationConfig.provider !== 'whisper' || this.communicationConfig.enabled !== true)) {
2017-12-28 12:40:11 -05:00
return;
}
2017-12-30 20:42:52 -05:00
// TODO: possible race condition could be a concern
this.events.request("version:get:web3", function(web3Version) {
let code = "";
2017-12-30 21:02:46 -05:00
code += "\n" + fs.readFileSync(utils.joinPath(__dirname, 'js', 'message_events.js')).toString();
2017-12-28 12:40:11 -05:00
2017-12-30 20:42:52 -05:00
if (web3Version[0] === "0") {
2017-12-30 21:02:46 -05:00
code += "\n" + fs.readFileSync(utils.joinPath(__dirname, 'js', 'embarkjs_old_web3.js')).toString();
2017-12-30 20:42:52 -05:00
code += "\nEmbarkJS.Messages.registerProvider('whisper', __embarkWhisperOld);";
} else {
2017-12-30 21:02:46 -05:00
code += "\n" + fs.readFileSync(utils.joinPath(__dirname, 'js', 'embarkjs.js')).toString();
2017-12-30 20:42:52 -05:00
code += "\nEmbarkJS.Messages.registerProvider('whisper', __embarkWhisperNewWeb3);";
}
self.embark.addCodeToEmbarkJS(code);
});
2017-12-28 12:40:11 -05:00
}
2017-12-28 18:10:43 -05:00
addSetProvider() {
let connection = this.communicationConfig.connection || {};
2018-08-28 14:43:52 +01:00
const shouldInit = (communicationConfig) => {
return (communicationConfig.provider === 'whisper' && communicationConfig.enabled === true);
};
2017-12-28 18:10:43 -05:00
// todo: make the add code a function as well
2018-08-28 14:43:52 +01:00
const config = {
2018-07-15 19:36:52 -05:00
server: canonicalHost(connection.host || defaultHost),
2017-12-28 18:10:43 -05:00
port: connection.port || '8546',
type: connection.type || 'ws'
2018-08-21 18:33:39 -05:00
};
2018-08-28 14:43:52 +01:00
const code = `\nEmbarkJS.Messages.setProvider('whisper', ${JSON.stringify(config)});`;
2017-12-28 18:10:43 -05:00
this.embark.addProviderInit('communication', code, shouldInit);
2018-08-28 14:43:52 +01:00
const consoleConfig = Object.assign({}, config, {providerOptions: {headers: {Origin: "http://embark"}}});
2018-08-28 14:43:52 +01:00
const consoleCode = `\nEmbarkJS.Messages.setProvider('whisper', ${JSON.stringify(consoleConfig)});`;
this.embark.addConsoleProviderInit('communication', consoleCode, shouldInit);
2017-12-28 18:10:43 -05:00
}
2017-12-28 12:40:11 -05:00
}
module.exports = Whisper;