forked from cerc-io/plugeth
482 lines
11 KiB
JavaScript
482 lines
11 KiB
JavaScript
// Copyright (c) 2013-2014, Jeffrey Wilcke. All rights reserved.
|
|
//
|
|
// This library is free software; you can redistribute it and/or
|
|
// modify it under the terms of the GNU General Public
|
|
// License as published by the Free Software Foundation; either
|
|
// version 2.1 of the License, or (at your option) any later version.
|
|
//
|
|
// This library is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
// General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this library; if not, write to the Free Software
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
|
// MA 02110-1301 USA
|
|
|
|
// The magic return variable. The magic return variable will be set during the execution of the QML call.
|
|
(function(window) {
|
|
var Promise = window.Promise;
|
|
if(typeof(Promise) === "undefined") {
|
|
var Promise = Q.Promise;
|
|
}
|
|
|
|
function isPromise(o) {
|
|
return typeof o === "object" && o.then
|
|
}
|
|
|
|
window.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";
|
|
}
|
|
|
|
postData({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 Q.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
params.data = params.data.join("");
|
|
postData({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) {
|
|
postData({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 Q.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
postData({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 Q.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
postData({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 Q.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
postData({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 Q.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
postData({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 Q.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
postData({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 Q.all(promises).then(function() {
|
|
return new Promise(function(resolve, reject) {
|
|
postData({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) {
|
|
postData({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) {
|
|
postData({call: "getCoinBase"}, function(coinbase) {
|
|
resolve(coinbase);
|
|
});
|
|
});
|
|
},
|
|
});
|
|
|
|
Object.defineProperty(eth, "listening", {
|
|
get: function() {
|
|
return new Promise(function(resolve, reject) {
|
|
postData({call: "getIsListening"}, function(listening) {
|
|
resolve(listening);
|
|
});
|
|
});
|
|
},
|
|
});
|
|
|
|
|
|
Object.defineProperty(eth, "mining", {
|
|
get: function() {
|
|
return new Promise(function(resolve, reject) {
|
|
postData({call: "getIsMining"}, function(mining) {
|
|
resolve(mining);
|
|
});
|
|
});
|
|
},
|
|
});
|
|
|
|
Object.defineProperty(eth, "peerCount", {
|
|
get: function() {
|
|
return new Promise(function(resolve, reject) {
|
|
postData({call: "getPeerCount"}, function(peerCount) {
|
|
resolve(peerCount);
|
|
});
|
|
});
|
|
},
|
|
});
|
|
|
|
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) {
|
|
postData({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) {
|
|
postData({call: "uninstallFilter", args:[id]});
|
|
});
|
|
};
|
|
|
|
Filter.prototype.messages = function() {
|
|
var self=this;
|
|
return Q.all([this.promise]).then(function() {
|
|
var id = self.id
|
|
return new Promise(function(resolve, reject) {
|
|
postData({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);
|
|
}
|
|
});
|
|
|
|
|
|
var g_seed = 1;
|
|
function postData(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, JSON.stringify(data))
|
|
}
|
|
})(this);
|