mirror of https://github.com/embarklabs/embark.git
Merge pull request #622 from embark-framework/features/ens-demo
Add ENS to Demo app and also add small fix
This commit is contained in:
commit
99af5f1817
14
js/embark.js
14
js/embark.js
|
@ -393,19 +393,23 @@ EmbarkJS.Names.setProvider = function (provider, options) {
|
||||||
};
|
};
|
||||||
|
|
||||||
// resolve resolves a name into an identifier of some kind
|
// resolve resolves a name into an identifier of some kind
|
||||||
EmbarkJS.Names.resolve = function (name) {
|
EmbarkJS.Names.resolve = function (name, callback) {
|
||||||
if (!this.currentNameSystems) {
|
if (!this.currentNameSystems) {
|
||||||
throw new Error('Name system provider not set; e.g EmbarkJS.Names.setProvider("ens")');
|
throw new Error('Name system provider not set; e.g EmbarkJS.Names.setProvider("ens")');
|
||||||
}
|
}
|
||||||
return this.currentNameSystems.resolve(name);
|
return this.currentNameSystems.resolve(name, callback);
|
||||||
};
|
};
|
||||||
|
|
||||||
// the reverse of resolve, resolves using an identifier to get to a name
|
// the reverse of resolve, resolves using an identifier to get to a name
|
||||||
EmbarkJS.Names.lookup = function (identifier) {
|
EmbarkJS.Names.lookup = function (identifier, callback) {
|
||||||
if (!this.currentNameSystems) {
|
if (!this.currentNameSystems) {
|
||||||
throw new Error('Name system provider not set; e.g EmbarkJS.Names.setProvider("ens")');
|
throw new Error('Name system provider not set; e.g EmbarkJS.Names.setProvider("ens")');
|
||||||
}
|
}
|
||||||
return this.currentNameSystems.lookup(identifier);
|
return this.currentNameSystems.lookup(identifier, callback);
|
||||||
|
};
|
||||||
|
|
||||||
|
EmbarkJS.Names.isAvailable = function () {
|
||||||
|
return this.currentNameSystems.isAvailable();
|
||||||
};
|
};
|
||||||
|
|
||||||
// To Implement
|
// To Implement
|
||||||
|
@ -417,7 +421,7 @@ EmbarkJS.Names.register = function(name, options) {
|
||||||
throw new Error('Name system provider not set; e.g EmbarkJS.Names.setProvider("ens")');
|
throw new Error('Name system provider not set; e.g EmbarkJS.Names.setProvider("ens")');
|
||||||
}
|
}
|
||||||
return this.currentNameSystems.register(name, options);
|
return this.currentNameSystems.register(name, options);
|
||||||
}
|
};
|
||||||
|
|
||||||
EmbarkJS.Utils = {
|
EmbarkJS.Utils = {
|
||||||
fromAscii: function (str) {
|
fromAscii: function (str) {
|
||||||
|
|
|
@ -21,7 +21,7 @@ class GethCommands {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (config.syncmode || config.syncMode) {
|
if (config.syncmode || config.syncMode) {
|
||||||
cmd.push("--syncmode=" + config.syncmode || config.syncMode);
|
cmd.push("--syncmode=" + (config.syncmode || config.syncMode));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (config.account && config.account.password) {
|
if (config.account && config.account.password) {
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
|
/*global EmbarkJS, web3*/
|
||||||
|
|
||||||
import namehash from 'eth-ens-namehash';
|
import namehash from 'eth-ens-namehash';
|
||||||
|
|
||||||
/*global EmbarkJS*/
|
|
||||||
let __embarkENS = {};
|
let __embarkENS = {};
|
||||||
|
|
||||||
// resolver interface
|
// resolver interface
|
||||||
|
@ -153,56 +154,90 @@ __embarkENS.resolverInterface = [
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
|
const providerNotSetError = 'ENS provider not set';
|
||||||
const NoDecodeAddrError = 'Error: Couldn\'t decode address from ABI: 0x';
|
const NoDecodeAddrError = 'Error: Couldn\'t decode address from ABI: 0x';
|
||||||
const NoDecodeStringError = 'ERROR: The returned value is not a convertible string: 0x0';
|
const NoDecodeStringError = 'ERROR: The returned value is not a convertible string: 0x0';
|
||||||
|
|
||||||
|
__embarkENS.registryAddresses = {
|
||||||
|
// Mainnet
|
||||||
|
"1": "0x314159265dd8dbb310642f98f50c066173c1259b",
|
||||||
|
// Ropsten
|
||||||
|
"3": "0x112234455c3a32fd11230c42e7bccd4a84e02010",
|
||||||
|
// Rinkeby
|
||||||
|
"4": "0xe7410170f87102DF0055eB195163A03B7F2Bff4A"
|
||||||
|
};
|
||||||
|
|
||||||
__embarkENS.setProvider = function (config) {
|
__embarkENS.setProvider = function (config) {
|
||||||
const self = this;
|
const self = this;
|
||||||
|
const ERROR_MESSAGE = 'ENS is not available in this chain';
|
||||||
EmbarkJS.onReady(() => {
|
EmbarkJS.onReady(() => {
|
||||||
self.ens = new EmbarkJS.Contract({abi: config.abi, address: config.address});
|
web3.eth.net.getId((err, id) => {
|
||||||
|
if (err) {
|
||||||
|
if (err.message.indexOf('Provider not set or invalid') > -1) {
|
||||||
|
console.warn(ERROR_MESSAGE);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
console.error(err);
|
||||||
|
}
|
||||||
|
if (!self.registryAddresses[id]) {
|
||||||
|
console.warn(ERROR_MESSAGE);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
self.isAvailable = true;
|
||||||
|
self.ens = new EmbarkJS.Contract({abi: config.abi, address: self.registryAddresses[id]});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
__embarkENS.resolve = function (name) {
|
__embarkENS.resolve = function (name, callback) {
|
||||||
const self = this;
|
callback = callback || function () {};
|
||||||
|
if (!this.ens) {
|
||||||
if (self.ens === undefined) return;
|
return callback(providerNotSetError);
|
||||||
|
|
||||||
let node = namehash.hash(name);
|
|
||||||
|
|
||||||
return self.ens.methods.resolver(node).call().then((resolverAddress) => {
|
|
||||||
let resolverContract = new EmbarkJS.Contract({abi: self.resolverInterface, address: resolverAddress});
|
|
||||||
return resolverContract.methods.addr(node).call();
|
|
||||||
}).then((addr) => {
|
|
||||||
return addr;
|
|
||||||
}).catch((err) => {
|
|
||||||
if (err === NoDecodeAddrError) {
|
|
||||||
console.log(name + " is not registered");
|
|
||||||
return "0x";
|
|
||||||
}
|
|
||||||
return err;
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
__embarkENS.lookup = function (address) {
|
|
||||||
const self = this;
|
|
||||||
|
|
||||||
if (!self.ens) {
|
|
||||||
console.log("ENS provider not set. Exiting.");
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
if (address.startsWith("0x")) address = address.slice(2);
|
let node = namehash.hash(name);
|
||||||
|
|
||||||
|
function cb(err, addr) {
|
||||||
|
if (err === NoDecodeAddrError) {
|
||||||
|
return callback(name + " is not registered", "0x");
|
||||||
|
}
|
||||||
|
callback(err, addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.ens.methods.resolver(node).call((err, resolverAddress) => {
|
||||||
|
if (err) {
|
||||||
|
return cb(err);
|
||||||
|
}
|
||||||
|
let resolverContract = new EmbarkJS.Contract({abi: this.resolverInterface, address: resolverAddress});
|
||||||
|
resolverContract.methods.addr(node).call(cb);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
__embarkENS.lookup = function (address, callback) {
|
||||||
|
callback = callback || function () {};
|
||||||
|
if (!this.ens) {
|
||||||
|
return callback(providerNotSetError);
|
||||||
|
}
|
||||||
|
if (address.startsWith("0x")) {
|
||||||
|
address = address.slice(2);
|
||||||
|
}
|
||||||
let node = namehash.hash(address.toLowerCase() + ".addr.reverse");
|
let node = namehash.hash(address.toLowerCase() + ".addr.reverse");
|
||||||
|
|
||||||
return self.ens.methods.resolver(node).call().then((resolverAddress) => {
|
function cb(err, name) {
|
||||||
let resolverContract = new EmbarkJS.Contract({abi: self.resolverInterface, address: resolverAddress});
|
|
||||||
return resolverContract.methods.name(node).call();
|
|
||||||
}).catch((err) => {
|
|
||||||
if (err === NoDecodeStringError || err === NoDecodeAddrError) {
|
if (err === NoDecodeStringError || err === NoDecodeAddrError) {
|
||||||
console.log('Address does not resolve to name. Try syncing chain.');
|
return callback('Address does not resolve to name. Try syncing chain.');
|
||||||
return "";
|
|
||||||
}
|
}
|
||||||
return err;
|
return callback(err, name);
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.ens.methods.resolver(node).call((err, resolverAddress) => {
|
||||||
|
if (err) {
|
||||||
|
return cb(err);
|
||||||
|
}
|
||||||
|
let resolverContract = new EmbarkJS.Contract({abi: this.resolverInterface, address: resolverAddress});
|
||||||
|
resolverContract.methods.name(node).call(cb);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
__embarkENS.isAvailable = function () {
|
||||||
|
return Boolean(this.isAvailable);
|
||||||
|
};
|
||||||
|
|
|
@ -0,0 +1,121 @@
|
||||||
|
import EmbarkJS from 'Embark/EmbarkJS';
|
||||||
|
import React from 'react';
|
||||||
|
import {Alert, Form, FormGroup, FormControl, Button} from 'react-bootstrap';
|
||||||
|
|
||||||
|
window.EmbarkJS = EmbarkJS;
|
||||||
|
|
||||||
|
class ENS extends React.Component {
|
||||||
|
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
|
||||||
|
this.state = {
|
||||||
|
valueResolve: 'ethereumfoundation.eth',
|
||||||
|
responseResolve: null,
|
||||||
|
isResolveError: false,
|
||||||
|
valueLookup: '0xfB6916095ca1df60bB79Ce92cE3Ea74c37c5d359',
|
||||||
|
responseLookup: null,
|
||||||
|
isLookupError: false,
|
||||||
|
embarkLogs: []
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
handleChange(stateName, e) {
|
||||||
|
this.setState({[stateName]: e.target.value});
|
||||||
|
}
|
||||||
|
|
||||||
|
resolveName(e) {
|
||||||
|
e.preventDefault();
|
||||||
|
const embarkLogs = this.state.embarkLogs;
|
||||||
|
embarkLogs.push(`EmbarkJS.Names.resolve('${this.state.valueResolve}', console.log)`);
|
||||||
|
|
||||||
|
this.setState({
|
||||||
|
embarkLogs: embarkLogs
|
||||||
|
});
|
||||||
|
EmbarkJS.Names.resolve(this.state.valueResolve, (err, result) => {
|
||||||
|
if (err) {
|
||||||
|
return this.setState({
|
||||||
|
responseResolve: err,
|
||||||
|
isResolveError: true
|
||||||
|
});
|
||||||
|
}
|
||||||
|
this.setState({
|
||||||
|
responseResolve: result,
|
||||||
|
isResolveError: false
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
lookupAddress(e) {
|
||||||
|
e.preventDefault();
|
||||||
|
const embarkLogs = this.state.embarkLogs;
|
||||||
|
embarkLogs.push(`EmbarkJS.Names.resolve('${this.state.valueLookup}', console.log)`);
|
||||||
|
|
||||||
|
this.setState({
|
||||||
|
embarkLogs: embarkLogs
|
||||||
|
});
|
||||||
|
EmbarkJS.Names.lookup(this.state.valueLookup, (err, result) => {
|
||||||
|
if (err) {
|
||||||
|
return this.setState({
|
||||||
|
responseLookup: err,
|
||||||
|
isLookupError: true
|
||||||
|
});
|
||||||
|
}
|
||||||
|
this.setState({
|
||||||
|
responseLookup: result,
|
||||||
|
isLookupError: false
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
return (<React.Fragment>
|
||||||
|
{
|
||||||
|
!this.props.enabled ?
|
||||||
|
<React.Fragment>
|
||||||
|
<Alert bsStyle="warning">ENS provider might not be set</Alert>
|
||||||
|
</React.Fragment> : ''
|
||||||
|
}
|
||||||
|
<h3>Resolve a name</h3>
|
||||||
|
<Form inline>
|
||||||
|
<FormGroup>
|
||||||
|
{this.state.responseResolve &&
|
||||||
|
<Alert className="alert-result" bsStyle={this.state.isResolveError ? 'danger' : 'success'}>
|
||||||
|
Resolved address: <span className="value">{this.state.responseResolve}</span>
|
||||||
|
</Alert>}
|
||||||
|
<FormControl
|
||||||
|
type="text"
|
||||||
|
defaultValue={this.state.valueResolve}
|
||||||
|
onChange={(e) => this.handleChange('valueResolve', e)}/>
|
||||||
|
<Button bsStyle="primary" onClick={(e) => this.resolveName(e)}>Resolve name</Button>
|
||||||
|
</FormGroup>
|
||||||
|
</Form>
|
||||||
|
|
||||||
|
<h3>Lookup an address</h3>
|
||||||
|
<Form inline>
|
||||||
|
<FormGroup>
|
||||||
|
{this.state.responseLookup &&
|
||||||
|
<Alert className="alert-result" bsStyle={this.state.isLookupError ? 'danger' : 'success'}>
|
||||||
|
Looked up domain: <span className="value">{this.state.responseLookup}</span>
|
||||||
|
</Alert>}
|
||||||
|
<FormControl
|
||||||
|
type="text"
|
||||||
|
defaultValue={this.state.valueLookup}
|
||||||
|
onChange={(e) => this.handleChange('valueLookup', e)}/>
|
||||||
|
<Button bsStyle="primary" onClick={(e) => this.lookupAddress(e)}>Lookup address</Button>
|
||||||
|
</FormGroup>
|
||||||
|
</Form>
|
||||||
|
|
||||||
|
<h3>Embark Calls </h3>
|
||||||
|
<p>Javascript calls being made: </p>
|
||||||
|
<div className="logs">
|
||||||
|
{
|
||||||
|
this.state.embarkLogs.map((item, i) => <p key={i}>{item}</p>)
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
</React.Fragment>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default ENS;
|
|
@ -50,4 +50,8 @@ div {
|
||||||
|
|
||||||
input.form-control {
|
input.form-control {
|
||||||
margin-right: 5px;
|
margin-right: 5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.alert-result {
|
||||||
|
margin-left: 0;
|
||||||
|
}
|
||||||
|
|
|
@ -1,11 +1,12 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import ReactDOM from 'react-dom';
|
import ReactDOM from 'react-dom';
|
||||||
import { Tabs, Tab } from 'react-bootstrap';
|
import {Tabs, Tab} from 'react-bootstrap';
|
||||||
|
|
||||||
import EmbarkJS from 'Embark/EmbarkJS';
|
import EmbarkJS from 'Embark/EmbarkJS';
|
||||||
import Blockchain from './components/blockchain';
|
import Blockchain from './components/blockchain';
|
||||||
import Whisper from './components/whisper';
|
import Whisper from './components/whisper';
|
||||||
import Storage from './components/storage';
|
import Storage from './components/storage';
|
||||||
|
import ENS from './components/ens';
|
||||||
|
|
||||||
import './dapp.css';
|
import './dapp.css';
|
||||||
|
|
||||||
|
@ -16,56 +17,61 @@ class App extends React.Component {
|
||||||
|
|
||||||
this.state = {
|
this.state = {
|
||||||
whisperEnabled: false,
|
whisperEnabled: false,
|
||||||
storageEnabled: false
|
storageEnabled: false,
|
||||||
}
|
ensEnabled: false
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidMount(){
|
componentDidMount() {
|
||||||
EmbarkJS.onReady(() => {
|
EmbarkJS.onReady(() => {
|
||||||
if (EmbarkJS.isNewWeb3()) {
|
if (EmbarkJS.isNewWeb3()) {
|
||||||
EmbarkJS.Messages.Providers.whisper.getWhisperVersion((err, version) => {
|
EmbarkJS.Messages.Providers.whisper.getWhisperVersion((err, _version) => {
|
||||||
if(!err)
|
if (err) {
|
||||||
this.setState({whisperEnabled: true})
|
return console.log(err);
|
||||||
else
|
}
|
||||||
console.log(err);
|
this.setState({whisperEnabled: true});
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
if (EmbarkJS.Messages.providerName === 'whisper') {
|
if (EmbarkJS.Messages.providerName === 'whisper') {
|
||||||
EmbarkJS.Messages.getWhisperVersion((err, version) => {
|
EmbarkJS.Messages.getWhisperVersion((err, _version) => {
|
||||||
if(!err)
|
if (err) {
|
||||||
this.setState({whisperEnabled: true})
|
return console.log(err);
|
||||||
else
|
}
|
||||||
console.log(err);
|
this.setState({whisperEnabled: true});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this.setState({
|
this.setState({
|
||||||
storageEnabled: true
|
storageEnabled: true,
|
||||||
|
ensEnabled: EmbarkJS.Names.isAvailable()
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
_renderStatus(title, available){
|
_renderStatus(title, available) {
|
||||||
let className = available ? 'pull-right status-online' : 'pull-right status-offline';
|
let className = available ? 'pull-right status-online' : 'pull-right status-offline';
|
||||||
return <React.Fragment>
|
return <React.Fragment>
|
||||||
{title}
|
{title}
|
||||||
<span className={className}></span>
|
<span className={className}></span>
|
||||||
</React.Fragment>;
|
</React.Fragment>;
|
||||||
}
|
}
|
||||||
|
|
||||||
render(){
|
render() {
|
||||||
return (<div><h3>Embark - Usage Example</h3>
|
return (<div><h3>Embark - Usage Example</h3>
|
||||||
<Tabs defaultActiveKey={1} id="uncontrolled-tab-example">
|
<Tabs defaultActiveKey={1} id="uncontrolled-tab-example">
|
||||||
<Tab eventKey={1} title="Blockchain">
|
<Tab eventKey={1} title="Blockchain">
|
||||||
<Blockchain />
|
<Blockchain/>
|
||||||
</Tab>
|
</Tab>
|
||||||
<Tab eventKey={2} title={this._renderStatus('Decentralized Storage', this.state.storageEnabled)}>
|
<Tab eventKey={2} title={this._renderStatus('Decentralized Storage', this.state.storageEnabled)}>
|
||||||
<Storage enabled={this.state.storageEnabled} />
|
<Storage enabled={this.state.storageEnabled}/>
|
||||||
</Tab>
|
</Tab>
|
||||||
<Tab eventKey={3} title={this._renderStatus('P2P communication (Whisper/Orbit)', this.state.whisperEnabled)}>
|
<Tab eventKey={3} title={this._renderStatus('P2P communication (Whisper)', this.state.whisperEnabled)}>
|
||||||
<Whisper enabled={this.state.whisperEnabled} />
|
<Whisper enabled={this.state.whisperEnabled}/>
|
||||||
|
</Tab>
|
||||||
|
<Tab eventKey={4} title={this._renderStatus('Naming (ENS)', this.state.ensEnabled)}>
|
||||||
|
<ENS enabled={this.state.ensEnabled}/>
|
||||||
</Tab>
|
</Tab>
|
||||||
</Tabs>
|
</Tabs>
|
||||||
</div>);
|
</div>);
|
||||||
|
|
Loading…
Reference in New Issue