3bc238b1ce
An ethereum provider provides an interface from which communication can be done.
584 lines
17 KiB
JavaScript
584 lines
17 KiB
JavaScript
(function(window) {
|
|
function isPromise(o) {
|
|
return typeof o === "object" && o.then
|
|
}
|
|
|
|
var eth = {
|
|
_callbacks: {},
|
|
_events: {},
|
|
prototype: Object(),
|
|
|
|
toHex: function(str) {
|
|
var hex = "";
|
|
for(var i = 0; i < str.length; i++) {
|
|
var n = str.charCodeAt(i).toString(16);
|
|
hex += n.length < 2 ? '0' + n : n;
|
|
}
|
|
|
|
return hex;
|
|
},
|
|
|
|
toAscii: function(hex) {
|
|
// Find termination
|
|
var str = "";
|
|
var i = 0, l = hex.length;
|
|
for(; i < l; i+=2) {
|
|
var code = hex.charCodeAt(i)
|
|
if(code == 0) {
|
|
break;
|
|
}
|
|
|
|
str += String.fromCharCode(parseInt(hex.substr(i, 2), 16));
|
|
}
|
|
|
|
return str;
|
|
},
|
|
|
|
fromAscii: function(str, pad) {
|
|
if(pad === undefined) {
|
|
pad = 32
|
|
}
|
|
|
|
var hex = this.toHex(str);
|
|
|
|
while(hex.length < pad*2)
|
|
hex += "00";
|
|
|
|
return hex
|
|
},
|
|
|
|
block: function(numberOrHash) {
|
|
return new Promise(function(resolve, reject) {
|
|
var func;
|
|
if(typeof numberOrHash == "string") {
|
|
func = "getBlockByHash";
|
|
} else {
|
|
func = "getBlockByNumber";
|
|
}
|
|
|
|
eth.provider.send({call: func, args: [numberOrHash]}, function(block) {
|
|
if(block)
|
|
resolve(block);
|
|
else
|
|
reject("not found");
|
|
|
|
});
|
|
});
|
|
},
|
|
|
|
transact: function(params) {
|
|
if(params === undefined) {
|
|
params = {};
|
|
}
|
|
|
|
if(params.endowment !== undefined)
|
|
params.value = params.endowment;
|
|
if(params.code !== undefined)
|
|
params.data = params.code;
|
|
|
|
|
|
var promises = []
|
|
if(isPromise(params.to)) {
|
|
promises.push(params.to.then(function(_to) { params.to = _to; }));
|
|
}
|
|
if(isPromise(params.from)) {
|
|
promises.push(params.from.then(function(_from) { params.from = _from; }));
|
|
}
|
|
|
|
if(typeof params.data !== "object" || isPromise(params.data)) {
|
|
params.data = [params.data]
|
|
}
|
|
|
|
var data = params.data;
|
|
for(var i = 0; i < params.data.length; i++) {
|
|
if(isPromise(params.data[i])) {
|
|
var promise = params.data[i];
|
|
var _i = i;
|
|
promises.push(promise.then(function(_arg) { params.data[_i] = _arg; }));
|
|
}
|
|
}
|
|
|
|
// Make sure everything is string
|
|
var fields = ["value", "gas", "gasPrice"];
|
|
for(var i = 0; i < fields.length; i++) {
|
|
if(params[fields[i]] === undefined) {
|
|
params[fields[i]] = "";
|
|
}
|
|
params[fields[i]] = params[fields[i]].toString();
|
|
}
|
|
|
|
// Load promises then call the last "transact".
|
|
return Promise.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
params.data = params.data.join("");
|
|
eth.provider.send({call: "transact", args: [params]}, function(data) {
|
|
if(data[1])
|
|
reject(data[0]);
|
|
else
|
|
resolve(data[0]);
|
|
});
|
|
});
|
|
})
|
|
},
|
|
|
|
compile: function(code) {
|
|
return new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: "compile", args: [code]}, function(data) {
|
|
if(data[1])
|
|
reject(data[0]);
|
|
else
|
|
resolve(data[0]);
|
|
});
|
|
});
|
|
},
|
|
|
|
balanceAt: function(address) {
|
|
var promises = [];
|
|
|
|
if(isPromise(address)) {
|
|
promises.push(address.then(function(_address) { address = _address; }));
|
|
}
|
|
|
|
return Promise.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: "getBalanceAt", args: [address]}, function(balance) {
|
|
resolve(balance);
|
|
});
|
|
});
|
|
});
|
|
},
|
|
|
|
countAt: function(address) {
|
|
var promises = [];
|
|
|
|
if(isPromise(address)) {
|
|
promises.push(address.then(function(_address) { address = _address; }));
|
|
}
|
|
|
|
return Promise.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: "getCountAt", args: [address]}, function(count) {
|
|
resolve(count);
|
|
});
|
|
});
|
|
});
|
|
},
|
|
|
|
codeAt: function(address) {
|
|
var promises = [];
|
|
|
|
if(isPromise(address)) {
|
|
promises.push(address.then(function(_address) { address = _address; }));
|
|
}
|
|
|
|
return Promise.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: "getCodeAt", args: [address]}, function(code) {
|
|
resolve(code);
|
|
});
|
|
});
|
|
});
|
|
},
|
|
|
|
storageAt: function(address, storageAddress) {
|
|
var promises = [];
|
|
|
|
if(isPromise(address)) {
|
|
promises.push(address.then(function(_address) { address = _address; }));
|
|
}
|
|
|
|
if(isPromise(storageAddress)) {
|
|
promises.push(storageAddress.then(function(_sa) { storageAddress = _sa; }));
|
|
}
|
|
|
|
return Promise.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: "getStorageAt", args: [address, storageAddress]}, function(entry) {
|
|
resolve(entry);
|
|
});
|
|
});
|
|
});
|
|
},
|
|
|
|
stateAt: function(address, storageAddress) {
|
|
return this.storageAt(address, storageAddress);
|
|
},
|
|
|
|
call: function(params) {
|
|
if(params === undefined) {
|
|
params = {};
|
|
}
|
|
|
|
if(params.endowment !== undefined)
|
|
params.value = params.endowment;
|
|
if(params.code !== undefined)
|
|
params.data = params.code;
|
|
|
|
|
|
var promises = []
|
|
if(isPromise(params.to)) {
|
|
promises.push(params.to.then(function(_to) { params.to = _to; }));
|
|
}
|
|
if(isPromise(params.from)) {
|
|
promises.push(params.from.then(function(_from) { params.from = _from; }));
|
|
}
|
|
|
|
if(isPromise(params.data)) {
|
|
promises.push(params.data.then(function(_code) { params.data = _code; }));
|
|
} else {
|
|
if(typeof params.data === "object") {
|
|
data = "";
|
|
for(var i = 0; i < params.data.length; i++) {
|
|
data += params.data[i]
|
|
}
|
|
} else {
|
|
data = params.data;
|
|
}
|
|
}
|
|
|
|
// Make sure everything is string
|
|
var fields = ["value", "gas", "gasPrice"];
|
|
for(var i = 0; i < fields.length; i++) {
|
|
if(params[fields[i]] === undefined) {
|
|
params[fields[i]] = "";
|
|
}
|
|
params[fields[i]] = params[fields[i]].toString();
|
|
}
|
|
|
|
// Load promises then call the last "transact".
|
|
return Promise.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: "call", args: params}, function(data) {
|
|
if(data[1])
|
|
reject(data[0]);
|
|
else
|
|
resolve(data[0]);
|
|
});
|
|
});
|
|
})
|
|
},
|
|
|
|
watch: function(params) {
|
|
return new Filter(params);
|
|
},
|
|
|
|
secretToAddress: function(key) {
|
|
var promises = [];
|
|
if(isPromise(key)) {
|
|
promises.push(key.then(function(_key) { key = _key; }));
|
|
}
|
|
|
|
return Promise.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: "getSecretToAddress", args: [key]}, function(address) {
|
|
resolve(address);
|
|
});
|
|
});
|
|
});
|
|
},
|
|
|
|
on: function(event, cb) {
|
|
if(eth._events[event] === undefined) {
|
|
eth._events[event] = [];
|
|
}
|
|
|
|
eth._events[event].push(cb);
|
|
|
|
return this
|
|
},
|
|
|
|
off: function(event, cb) {
|
|
if(eth._events[event] !== undefined) {
|
|
var callbacks = eth._events[event];
|
|
for(var i = 0; i < callbacks.length; i++) {
|
|
if(callbacks[i] === cb) {
|
|
delete callbacks[i];
|
|
}
|
|
}
|
|
}
|
|
|
|
return this
|
|
},
|
|
|
|
trigger: function(event, data) {
|
|
var callbacks = eth._events[event];
|
|
if(callbacks !== undefined) {
|
|
for(var i = 0; i < callbacks.length; i++) {
|
|
// Figure out whether the returned data was an array
|
|
// array means multiple return arguments (multiple params)
|
|
if(data instanceof Array) {
|
|
callbacks[i].apply(this, data);
|
|
} else {
|
|
callbacks[i].call(this, data);
|
|
}
|
|
}
|
|
}
|
|
},
|
|
};
|
|
|
|
// Eth object properties
|
|
Object.defineProperty(eth, "key", {
|
|
get: function() {
|
|
return new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: "getKey"}, function(k) {
|
|
resolve(k);
|
|
});
|
|
});
|
|
},
|
|
});
|
|
|
|
Object.defineProperty(eth, "gasPrice", {
|
|
get: function() {
|
|
return "10000000000000"
|
|
}
|
|
});
|
|
|
|
Object.defineProperty(eth, "coinbase", {
|
|
get: function() {
|
|
return new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: "getCoinBase"}, function(coinbase) {
|
|
resolve(coinbase);
|
|
});
|
|
});
|
|
},
|
|
});
|
|
|
|
Object.defineProperty(eth, "listening", {
|
|
get: function() {
|
|
return new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: "getIsListening"}, function(listening) {
|
|
resolve(listening);
|
|
});
|
|
});
|
|
},
|
|
});
|
|
|
|
|
|
Object.defineProperty(eth, "mining", {
|
|
get: function() {
|
|
return new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: "getIsMining"}, function(mining) {
|
|
resolve(mining);
|
|
});
|
|
});
|
|
},
|
|
});
|
|
|
|
Object.defineProperty(eth, "peerCount", {
|
|
get: function() {
|
|
return new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: "getPeerCount"}, function(peerCount) {
|
|
resolve(peerCount);
|
|
});
|
|
});
|
|
},
|
|
});
|
|
|
|
var ProviderManager = function() {
|
|
this.queued = [];
|
|
this.ready = false;
|
|
this.provider = undefined;
|
|
this.seed = 1;
|
|
};
|
|
ProviderManager.prototype.send = function(data, cb) {
|
|
data.seed = this.seed;
|
|
if(cb) {
|
|
eth._callbacks[data.seed] = cb;
|
|
}
|
|
|
|
if(data.args === undefined) {
|
|
data.args = [];
|
|
}
|
|
|
|
this.seed++;
|
|
|
|
if(this.provider !== undefined) {
|
|
this.provider.send(data);
|
|
} else {
|
|
this.queued = data;
|
|
}
|
|
};
|
|
|
|
ProviderManager.prototype.set = function(provider) {
|
|
if(this.provider !== undefined && this.provider.unload !== undefined) {
|
|
this.provider.unload();
|
|
}
|
|
|
|
this.provider = provider;
|
|
this.ready = true;
|
|
};
|
|
|
|
ProviderManager.prototype.sendQueued = function() {
|
|
for(var i = 0; this.queued.length; i++) {
|
|
// Resend
|
|
this.send(this.queued[i]);
|
|
}
|
|
};
|
|
|
|
ProviderManager.prototype.installed = function() {
|
|
return this.provider !== undefined;
|
|
};
|
|
|
|
eth.provider = new ProviderManager();
|
|
|
|
eth.setProvider = function(provider) {
|
|
eth.provider.set(provider);
|
|
|
|
eth.provider.sendQueued();
|
|
};
|
|
|
|
var EthWebSocket = function(host) {
|
|
// onmessage handlers
|
|
this.handlers = [];
|
|
// queue will be filled with messages if send is invoked before the ws is ready
|
|
this.queued = [];
|
|
this.ready = false;
|
|
|
|
this.ws = new WebSocket(host);
|
|
|
|
var self = this;
|
|
this.ws.onmessage = function(event) {
|
|
for(var i = 0; i < self.handlers.length; i++) {
|
|
self.handlers[i].call(self, event.data, event)
|
|
}
|
|
};
|
|
|
|
this.ws.onopen = function() {
|
|
self.ready = true;
|
|
|
|
for(var i = 0; i < self.queued.length; i++) {
|
|
// Resend
|
|
self.send(self.queued[i]);
|
|
}
|
|
};
|
|
};
|
|
|
|
EthWebSocket.prototype.send = function(jsonData) {
|
|
if(this.ready) {
|
|
var data = JSON.stringify(jsonData);
|
|
|
|
this.ws.send(data);
|
|
} else {
|
|
this.queued.push(jsonData);
|
|
}
|
|
};
|
|
|
|
EthWebSocket.prototype.onMessage = function(handler) {
|
|
this.handlers.push(handler);
|
|
};
|
|
|
|
EthWebSocket.prototype.unload = function() {
|
|
this.ws.close();
|
|
};
|
|
eth.WebSocket = EthWebSocket;
|
|
|
|
var filters = [];
|
|
var Filter = function(options) {
|
|
filters.push(this);
|
|
|
|
this.callbacks = [];
|
|
this.options = options;
|
|
|
|
var call;
|
|
if(options === "chain") {
|
|
call = "newFilterString"
|
|
} else if(typeof options === "object") {
|
|
call = "newFilter"
|
|
}
|
|
|
|
var self = this; // Cheaper than binding
|
|
this.promise = new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: call, args: [options]}, function(id) {
|
|
self.id = id;
|
|
|
|
resolve(id);
|
|
});
|
|
});
|
|
};
|
|
|
|
Filter.prototype.changed = function(callback) {
|
|
var self = this;
|
|
this.promise.then(function(id) {
|
|
self.callbacks.push(callback);
|
|
});
|
|
};
|
|
|
|
Filter.prototype.trigger = function(messages, id) {
|
|
if(id == this.id) {
|
|
for(var i = 0; i < this.callbacks.length; i++) {
|
|
this.callbacks[i].call(this, messages);
|
|
}
|
|
}
|
|
};
|
|
|
|
Filter.prototype.uninstall = function() {
|
|
this.promise.then(function(id) {
|
|
eth.provider.send({call: "uninstallFilter", args:[id]});
|
|
});
|
|
};
|
|
|
|
Filter.prototype.messages = function() {
|
|
var self=this;
|
|
return Promise.all([this.promise]).then(function() {
|
|
var id = self.id
|
|
return new Promise(function(resolve, reject) {
|
|
eth.provider.send({call: "getMessages", args: [id]}, function(messages) {
|
|
resolve(messages);
|
|
});
|
|
});
|
|
});
|
|
};
|
|
|
|
// Register to the messages callback. "messages" will be emitted when new messages
|
|
// from the client have been created.
|
|
eth.on("messages", function(messages, id) {
|
|
for(var i = 0; i < filters.length; i++) {
|
|
filters[i].trigger(messages, id);
|
|
}
|
|
});
|
|
|
|
|
|
// Install default provider
|
|
if(!eth.provider.installed()) {
|
|
var sock = new eth.WebSocket("ws://localhost:40404/eth");
|
|
sock.onMessage(function(ev) {
|
|
var data = JSON.parse(ev)
|
|
|
|
if(data._event !== undefined) {
|
|
eth.trigger(data._event, data.data);
|
|
} else {
|
|
if(data.seed) {
|
|
var cb = eth._callbacks[data.seed];
|
|
if(cb) {
|
|
cb.call(this, data.data)
|
|
|
|
// Remove the "trigger" callback
|
|
delete eth._callbacks[ev.seed];
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
eth.setProvider(sock);
|
|
}
|
|
|
|
window.eth = eth;
|
|
})(this);
|
|
|
|
/*
|
|
function eth.provider.send(data, cb) {
|
|
data.seed = g_seed;
|
|
if(cb) {
|
|
eth._callbacks[data.seed] = cb;
|
|
}
|
|
|
|
if(data.args === undefined) {
|
|
data.args = [];
|
|
}
|
|
|
|
g_seed++;
|
|
|
|
window._messagingAdapter.call(this, data)
|
|
}
|
|
*/
|