141 lines
4.5 KiB
JavaScript
Raw Normal View History

import EmbarkJS from 'Embark/EmbarkJS';
import React from 'react';
import {Alert, Form, FormGroup, Input, Button, FormText} from 'reactstrap';
2018-05-02 12:24:16 -04:00
class Whisper extends React.Component {
2018-05-02 12:24:16 -04:00
constructor (props) {
super(props);
2018-05-02 12:24:16 -04:00
this.state = {
listenTo: '',
channel: '',
message: '',
subscribedChannels: [],
channelIsValid: false,
listenToChannelIsValid: false,
2018-05-02 12:24:16 -04:00
messageList: [],
logs: []
};
}
2018-05-02 12:24:16 -04:00
handleChange (e, name) {
this.state[name] = e.target.value;
if (name === 'listenTo') {
this.state.listenToChannelIsValid = this.isChannelValid(e.target.value);
} else if (name === "channel") {
this.state.channelIsValid = this.isChannelValid(e.target.value);
}
2018-05-02 12:24:16 -04:00
this.setState(this.state);
}
2018-08-15 13:15:40 -04:00
checkEnter(e, func) {
if (e.key !== 'Enter') {
return;
}
e.preventDefault();
func.apply(this, [e]);
}
2018-05-02 12:24:16 -04:00
sendMessage (e) {
e.preventDefault();
EmbarkJS.Messages.sendMessage({topic: this.state.channel, data: this.state.message});
this.addToLog("EmbarkJS.Messages.sendMessage({topic: '" + this.state.channel + "', data: '" + this.state.message + "'})");
}
2018-05-02 12:24:16 -04:00
listenToChannel (e) {
e.preventDefault();
2018-05-02 12:24:16 -04:00
const subscribedChannels = this.state.subscribedChannels;
subscribedChannels.push(this.state.listenTo);
2018-05-02 12:24:16 -04:00
this.setState({
subscribedChannels
});
refactor: be consistent with callbacks and promises If a function receives a callback argument then it should not return a promise if the caller's callback will be invoked. Both invoking a callback and returning a promise can lead to at best confusion (in code review and at runtime) and at worst non-deterministic behavior, such as race conditions. Also, a caller supplying a callback may not handle a returned promise, leading to unhandled rejection errors. Refactor all readily identified functions where a callback argument can be supplied but the function returns a promise regardless. Make use of `callbackify` and `promisify` where it made sense to do so during the refactoring. Some callsites of the revised functions may have been accidentally overlooked and still need to be updated. Some functions that take callback arguments may execute them synchronously, at odds with control flow of a returned promise (if a callback wasn't supplied). Such cases should be identified and fixed so that asynchronous behavior is fully consistent whether the caller supplies a callback or receives a promise. Make sure promises that pass control flow to a callback ignore rejections, since those should be handled by the callback. Don't return promise instances unnecessarily from async functions (since they always return promises) and change some functions that return promises to async functions (where it's simple to do so). Whisper was using an ad hoc promise-like `messageEvents` object. However, that object behaved more like an observable, since promises either resolve or reject, and only do so one time. `messageEvents` was also intertwined with callbacks. Replace `messageEvents` with RxJS Observable. `listenTo` now returns Observable instances and callers can subscribe to them. `Blockchain.connect` of embarkjs could suffer from a race condition where tasks associated with `execWhenReady` might be ongoing when `connect`'s returned promise resolves/rejects (or a caller supplied callback fires). Attempt to ensure that returned-promise / supplied-callback control flow proceeds only after `execWhenReady` tasks have finished. The control flow involved is... rather involved, and it could use some further review and refactoring. Bump webpack and the hard-source-plugin for webpack. [util]: https://www.npmjs.com/package/util
2019-02-11 00:08:01 -06:00
const messageList = this.state.messageList;
EmbarkJS.Messages.listenTo({topic: [this.state.listenTo]}).subscribe(
message => {
2018-05-02 12:24:16 -04:00
messageList.push(<span>Channel: <b>{message.topic}</b> | Message: <b>{message.data}</b></span>);
refactor: be consistent with callbacks and promises If a function receives a callback argument then it should not return a promise if the caller's callback will be invoked. Both invoking a callback and returning a promise can lead to at best confusion (in code review and at runtime) and at worst non-deterministic behavior, such as race conditions. Also, a caller supplying a callback may not handle a returned promise, leading to unhandled rejection errors. Refactor all readily identified functions where a callback argument can be supplied but the function returns a promise regardless. Make use of `callbackify` and `promisify` where it made sense to do so during the refactoring. Some callsites of the revised functions may have been accidentally overlooked and still need to be updated. Some functions that take callback arguments may execute them synchronously, at odds with control flow of a returned promise (if a callback wasn't supplied). Such cases should be identified and fixed so that asynchronous behavior is fully consistent whether the caller supplies a callback or receives a promise. Make sure promises that pass control flow to a callback ignore rejections, since those should be handled by the callback. Don't return promise instances unnecessarily from async functions (since they always return promises) and change some functions that return promises to async functions (where it's simple to do so). Whisper was using an ad hoc promise-like `messageEvents` object. However, that object behaved more like an observable, since promises either resolve or reject, and only do so one time. `messageEvents` was also intertwined with callbacks. Replace `messageEvents` with RxJS Observable. `listenTo` now returns Observable instances and callers can subscribe to them. `Blockchain.connect` of embarkjs could suffer from a race condition where tasks associated with `execWhenReady` might be ongoing when `connect`'s returned promise resolves/rejects (or a caller supplied callback fires). Attempt to ensure that returned-promise / supplied-callback control flow proceeds only after `execWhenReady` tasks have finished. The control flow involved is... rather involved, and it could use some further review and refactoring. Bump webpack and the hard-source-plugin for webpack. [util]: https://www.npmjs.com/package/util
2019-02-11 00:08:01 -06:00
this.setState({messageList});
},
error => {
messageList.push(<span className="alert-danger">Error: {error.message || "Unknown Error"}</span>);
refactor: be consistent with callbacks and promises If a function receives a callback argument then it should not return a promise if the caller's callback will be invoked. Both invoking a callback and returning a promise can lead to at best confusion (in code review and at runtime) and at worst non-deterministic behavior, such as race conditions. Also, a caller supplying a callback may not handle a returned promise, leading to unhandled rejection errors. Refactor all readily identified functions where a callback argument can be supplied but the function returns a promise regardless. Make use of `callbackify` and `promisify` where it made sense to do so during the refactoring. Some callsites of the revised functions may have been accidentally overlooked and still need to be updated. Some functions that take callback arguments may execute them synchronously, at odds with control flow of a returned promise (if a callback wasn't supplied). Such cases should be identified and fixed so that asynchronous behavior is fully consistent whether the caller supplies a callback or receives a promise. Make sure promises that pass control flow to a callback ignore rejections, since those should be handled by the callback. Don't return promise instances unnecessarily from async functions (since they always return promises) and change some functions that return promises to async functions (where it's simple to do so). Whisper was using an ad hoc promise-like `messageEvents` object. However, that object behaved more like an observable, since promises either resolve or reject, and only do so one time. `messageEvents` was also intertwined with callbacks. Replace `messageEvents` with RxJS Observable. `listenTo` now returns Observable instances and callers can subscribe to them. `Blockchain.connect` of embarkjs could suffer from a race condition where tasks associated with `execWhenReady` might be ongoing when `connect`'s returned promise resolves/rejects (or a caller supplied callback fires). Attempt to ensure that returned-promise / supplied-callback control flow proceeds only after `execWhenReady` tasks have finished. The control flow involved is... rather involved, and it could use some further review and refactoring. Bump webpack and the hard-source-plugin for webpack. [util]: https://www.npmjs.com/package/util
2019-02-11 00:08:01 -06:00
this.setState({messageList});
2018-05-02 12:24:16 -04:00
}
refactor: be consistent with callbacks and promises If a function receives a callback argument then it should not return a promise if the caller's callback will be invoked. Both invoking a callback and returning a promise can lead to at best confusion (in code review and at runtime) and at worst non-deterministic behavior, such as race conditions. Also, a caller supplying a callback may not handle a returned promise, leading to unhandled rejection errors. Refactor all readily identified functions where a callback argument can be supplied but the function returns a promise regardless. Make use of `callbackify` and `promisify` where it made sense to do so during the refactoring. Some callsites of the revised functions may have been accidentally overlooked and still need to be updated. Some functions that take callback arguments may execute them synchronously, at odds with control flow of a returned promise (if a callback wasn't supplied). Such cases should be identified and fixed so that asynchronous behavior is fully consistent whether the caller supplies a callback or receives a promise. Make sure promises that pass control flow to a callback ignore rejections, since those should be handled by the callback. Don't return promise instances unnecessarily from async functions (since they always return promises) and change some functions that return promises to async functions (where it's simple to do so). Whisper was using an ad hoc promise-like `messageEvents` object. However, that object behaved more like an observable, since promises either resolve or reject, and only do so one time. `messageEvents` was also intertwined with callbacks. Replace `messageEvents` with RxJS Observable. `listenTo` now returns Observable instances and callers can subscribe to them. `Blockchain.connect` of embarkjs could suffer from a race condition where tasks associated with `execWhenReady` might be ongoing when `connect`'s returned promise resolves/rejects (or a caller supplied callback fires). Attempt to ensure that returned-promise / supplied-callback control flow proceeds only after `execWhenReady` tasks have finished. The control flow involved is... rather involved, and it could use some further review and refactoring. Bump webpack and the hard-source-plugin for webpack. [util]: https://www.npmjs.com/package/util
2019-02-11 00:08:01 -06:00
);
2018-05-02 12:24:16 -04:00
this.addToLog("EmbarkJS.Messages.listenTo({topic: ['" + this.state.listenTo + "']}).then(function(message) {})");
}
2018-05-02 12:24:16 -04:00
addToLog (txt) {
this.state.logs.push(txt);
this.setState({logs: this.state.logs});
}
isChannelValid(name) {
return name.length >= 4;
}
2018-05-02 12:24:16 -04:00
render () {
return (
<React.Fragment>
{
!this.props.enabled &&
2018-05-02 12:24:16 -04:00
<React.Fragment>
<Alert color="warning">The node you are using does not support Whisper</Alert>
<Alert color="warning">The node uses an unsupported version of Whisper</Alert>
</React.Fragment>}
2018-05-02 12:24:16 -04:00
<h3>Listen To channel</h3>
<Form onKeyDown={(e) => this.checkEnter(e, this.listenToChannel)}>
<FormGroup className="inline-input-btn">
<Input
2018-05-02 12:24:16 -04:00
type="text"
defaultValue={this.state.listenTo}
placeholder="channel"
onChange={e => this.handleChange(e, 'listenTo')}/>
<Button disabled={!this.state.listenToChannelIsValid} color="primary" onClick={(e) => this.listenToChannel(e)}>Start Listening</Button>
{!this.state.listenToChannelIsValid && <FormText color="danger">Channel has to be at least 4 characters long</FormText>}
2018-05-02 12:24:16 -04:00
<div id="subscribeList">
{this.state.subscribedChannels.map((item, i) => <p key={i}><span>Subscribed to <b>{item}</b>. Now try sending a message</span></p>)}
2018-05-02 12:24:16 -04:00
</div>
<h5>Messages received:</h5>
2018-05-02 12:24:16 -04:00
<div id="messagesList">
{this.state.messageList.map((item, i) => <p key={i}>{item}</p>)}
</div>
</FormGroup>
</Form>
2018-05-02 12:24:16 -04:00
<h3>Send Message</h3>
2018-08-15 13:15:40 -04:00
<Form inline onKeyDown={(e) => this.checkEnter(e, this.sendMessage)}>
2018-05-02 12:24:16 -04:00
<FormGroup>
<Input
2018-05-02 12:24:16 -04:00
type="text"
defaultValue={this.state.channel}
placeholder="channel"
onChange={e => this.handleChange(e, 'channel')}/>
<div className="inline-input-btn m-0">
<Input
type="text"
defaultValue={this.state.message}
placeholder="message"
onChange={e => this.handleChange(e, 'message')}/>
<Button color="primary" disabled={!this.state.channelIsValid} onClick={(e) => this.sendMessage(e)}>Send
Message</Button>
</div>
2018-05-02 12:24:16 -04:00
</FormGroup>
</Form>
2018-05-02 12:24:16 -04:00
<p>Javascript calls being made: </p>
<div className="logs">
<p>EmbarkJS.Messages.setProvider('whisper')</p>
{
this.state.logs.map((item, i) => <p key={i}>{item}</p>)
}
</div>
</React.Fragment>
);
}
}
export default Whisper;