mirror of
https://github.com/status-im/status-react.git
synced 2025-01-10 02:56:07 +00:00
Jakub Sokołowski
1f7fd17ff1
This way the name of the repo makes at least some sense and matches the `status-desktop` repo naming. Also updated `status-jenkins-lib` since it also contained references to `status-react` repo and job names. Signed-off-by: Jakub Sokołowski <jakub@status.im>
320 lines
11 KiB
JavaScript
320 lines
11 KiB
JavaScript
(function(){
|
|
if(typeof EthereumProvider === "undefined"){
|
|
var callbackId = 0;
|
|
var callbacks = {};
|
|
|
|
var bridgeSend = function (data) {
|
|
ReactNativeWebView.postMessage(JSON.stringify(data));
|
|
}
|
|
|
|
var history = window.history;
|
|
var pushState = history.pushState;
|
|
history.pushState = function(state) {
|
|
setTimeout(function () {
|
|
bridgeSend({
|
|
type: 'history-state-changed',
|
|
navState: { url: location.href, title: document.title }
|
|
});
|
|
}, 100);
|
|
return pushState.apply(history, arguments);
|
|
};
|
|
|
|
function sendAPIrequest(permission, params) {
|
|
var messageId = callbackId++;
|
|
var params = params || {};
|
|
|
|
bridgeSend({
|
|
type: 'api-request',
|
|
permission: permission,
|
|
messageId: messageId,
|
|
params: params
|
|
});
|
|
|
|
return new Promise(function (resolve, reject) {
|
|
params['resolve'] = resolve;
|
|
params['reject'] = reject;
|
|
callbacks[messageId] = params;
|
|
});
|
|
}
|
|
|
|
function qrCodeResponse(data, callback){
|
|
var result = data.data;
|
|
var regex = new RegExp(callback.regex);
|
|
if (!result) {
|
|
if (callback.reject) {
|
|
callback.reject(new Error("Cancelled"));
|
|
}
|
|
}
|
|
else if (regex.test(result)) {
|
|
if (callback.resolve) {
|
|
callback.resolve(result);
|
|
}
|
|
} else {
|
|
if (callback.reject) {
|
|
callback.reject(new Error("Doesn't match"));
|
|
}
|
|
}
|
|
}
|
|
|
|
function Unauthorized() {
|
|
this.name = "Unauthorized";
|
|
this.id = 4100;
|
|
this.code = 4100;
|
|
this.message = "The requested method and/or account has not been authorized by the user.";
|
|
}
|
|
Unauthorized.prototype = Object.create(Error.prototype);
|
|
|
|
function UserRejectedRequest() {
|
|
this.name = "UserRejectedRequest";
|
|
this.id = 4001;
|
|
this.code = 4001;
|
|
this.message = "The user rejected the request.";
|
|
}
|
|
UserRejectedRequest.prototype = Object.create(Error.prototype);
|
|
ReactNativeWebView.onMessage = function (message)
|
|
{
|
|
data = JSON.parse(message);
|
|
var id = data.messageId;
|
|
var callback = callbacks[id];
|
|
|
|
if (callback) {
|
|
if (data.type === "api-response") {
|
|
if (data.permission == 'qr-code'){
|
|
qrCodeResponse(data, callback);
|
|
} else if (data.isAllowed) {
|
|
if (data.permission == 'web3') {
|
|
var selectedAddress = data.data[0]
|
|
window.statusAppcurrentAccountAddress = selectedAddress;
|
|
// Set deprecated metamask fields
|
|
window.ethereum.selectedAddress = selectedAddress;
|
|
window.ethereum.emit("accountsChanged", data.data);
|
|
}
|
|
callback.resolve(data.data);
|
|
} else {
|
|
callback.reject(new UserRejectedRequest());
|
|
}
|
|
}
|
|
else if (data.type === "web3-send-async-callback")
|
|
{
|
|
if (callback.beta)
|
|
{
|
|
if (data.error)
|
|
{
|
|
if (data.error.code == 4100)
|
|
callback.reject(new Unauthorized());
|
|
else
|
|
callback.reject(data.error);
|
|
}
|
|
else
|
|
{
|
|
if (window.statusAppDebug) { console.log("resolve " + callback.method + " :" + JSON.stringify(data.result.result)); }
|
|
callback.resolve(data.result.result);
|
|
}
|
|
}
|
|
else if (callback.results)
|
|
{
|
|
callback.results.push(data.error || data.result);
|
|
if (callback.results.length == callback.num)
|
|
callback.callback(undefined, callback.results);
|
|
}
|
|
else
|
|
{
|
|
callback.callback(data.error, data.result);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
function web3Response (payload, result){
|
|
return {id: payload.id,
|
|
jsonrpc: "2.0",
|
|
result: result};
|
|
}
|
|
|
|
function getSyncResponse (payload) {
|
|
if (payload.method == "eth_accounts" && (typeof window.statusAppcurrentAccountAddress !== "undefined")) {
|
|
return web3Response(payload, [window.statusAppcurrentAccountAddress])
|
|
} else if (payload.method == "eth_coinbase" && (typeof window.statusAppcurrentAccountAddress !== "undefined")) {
|
|
return web3Response(payload, window.statusAppcurrentAccountAddress)
|
|
} else if (payload.method == "net_version"){
|
|
return web3Response(payload, window.statusAppNetworkId)
|
|
} else if (payload.method == "eth_chainId"){
|
|
return web3Response(payload, "0x" + Number(window.statusAppNetworkId).toString(16))
|
|
} else if (payload.method == "eth_uninstallFilter"){
|
|
return web3Response(payload, true);
|
|
} else {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
var StatusAPI = function () {};
|
|
|
|
StatusAPI.prototype.getContactCode = function () {
|
|
return sendAPIrequest('contact-code');
|
|
};
|
|
|
|
var EthereumProvider = function () {};
|
|
|
|
EthereumProvider.prototype.isStatus = true;
|
|
EthereumProvider.prototype.status = new StatusAPI();
|
|
EthereumProvider.prototype.isConnected = function () { return true; };
|
|
// Set legacy metamask fields https://docs.metamask.io/guide/ethereum-provider.html#legacy-api
|
|
EthereumProvider.prototype.networkVersion = window.statusAppNetworkId;
|
|
EthereumProvider.prototype.chainId = "0x" + Number(window.statusAppNetworkId).toString(16);
|
|
|
|
EthereumProvider.prototype._events = {};
|
|
|
|
EthereumProvider.prototype.on = function(name, listener) {
|
|
if (!this._events[name]) {
|
|
this._events[name] = [];
|
|
}
|
|
this._events[name].push(listener);
|
|
}
|
|
|
|
EthereumProvider.prototype.removeListener = function (name, listenerToRemove) {
|
|
if (!this._events[name]) {
|
|
return
|
|
}
|
|
|
|
const filterListeners = (listener) => listener !== listenerToRemove;
|
|
this._events[name] = this._events[name].filter(filterListeners);
|
|
}
|
|
|
|
EthereumProvider.prototype.removeAllListeners = function () {
|
|
this._events = {};
|
|
}
|
|
|
|
EthereumProvider.prototype.emit = function (name, data) {
|
|
if (!this._events[name]) {
|
|
return
|
|
}
|
|
this._events[name].forEach(cb => {
|
|
// Fixes: https://github.com/status-im/status-mobile/issues/13642
|
|
// Metamask also errors on the same issue, but it's using https://github.com/MetaMask/safe-event-emitter and therefore the dapp still works
|
|
try{
|
|
cb(data)
|
|
} catch(e){
|
|
setTimeout((()=>{throw e}))
|
|
}});
|
|
}
|
|
EthereumProvider.prototype.enable = function () {
|
|
if (window.statusAppDebug) { console.log("enable"); }
|
|
return sendAPIrequest('web3');
|
|
};
|
|
|
|
EthereumProvider.prototype.scanQRCode = function (regex) {
|
|
return sendAPIrequest('qr-code', {regex: regex});
|
|
};
|
|
|
|
EthereumProvider.prototype.request = function (requestArguments)
|
|
{
|
|
if (window.statusAppDebug) { console.log("request: " + JSON.stringify(requestArguments)); }
|
|
if (!requestArguments) {
|
|
return new Error('Request is not valid.');
|
|
}
|
|
var method = requestArguments.method;
|
|
|
|
if (!method) {
|
|
return new Error('Request is not valid.');
|
|
}
|
|
|
|
//Support for legacy send method
|
|
if (typeof method !== 'string') {
|
|
return this.sendSync(method);
|
|
}
|
|
|
|
if (method == 'eth_requestAccounts'){
|
|
return sendAPIrequest('web3');
|
|
}
|
|
|
|
var syncResponse = getSyncResponse({method: method});
|
|
if (syncResponse){
|
|
return new Promise(function (resolve, reject) {
|
|
if (window.statusAppDebug) { console.log("resolve sync: " + resolve + " " + JSON.stringify(syncResponse.result)); }
|
|
resolve(syncResponse.result);
|
|
});
|
|
}
|
|
|
|
var messageId = callbackId++;
|
|
var payload = {id: messageId,
|
|
jsonrpc: "2.0",
|
|
method: method,
|
|
params: requestArguments.params};
|
|
|
|
bridgeSend({type: 'web3-send-async-read-only',
|
|
messageId: messageId,
|
|
payload: payload});
|
|
|
|
return new Promise(function (resolve, reject) {
|
|
callbacks[messageId] = {beta: true,
|
|
method: method,
|
|
resolve: resolve,
|
|
reject: reject};
|
|
});
|
|
};
|
|
|
|
// (DEPRECATED) Support for legacy send method
|
|
EthereumProvider.prototype.send = function (method, params = [])
|
|
{
|
|
if (window.statusAppDebug) { console.log("send (legacy): " + method);}
|
|
return this.request({method: method, params: params});
|
|
}
|
|
|
|
// (DEPRECATED) Support for legacy sendSync method
|
|
EthereumProvider.prototype.sendSync = function (payload)
|
|
{
|
|
if (window.statusAppDebug) { console.log("sendSync (legacy)" + JSON.stringify(payload));}
|
|
if (payload.method == "eth_uninstallFilter"){
|
|
this.sendAsync(payload, function (res, err) {})
|
|
}
|
|
var syncResponse = getSyncResponse(payload);
|
|
if (syncResponse){
|
|
return syncResponse;
|
|
} else {
|
|
return web3Response(payload, null);
|
|
}
|
|
};
|
|
|
|
// (DEPRECATED) Support for legacy sendAsync method
|
|
EthereumProvider.prototype.sendAsync = function (payload, callback)
|
|
{
|
|
if (window.statusAppDebug) { console.log("sendAsync (legacy)" + JSON.stringify(payload));}
|
|
if (!payload) {
|
|
return new Error('Request is not valid.');
|
|
}
|
|
if (payload.method == 'eth_requestAccounts'){
|
|
return sendAPIrequest('web3');
|
|
}
|
|
var syncResponse = getSyncResponse(payload);
|
|
if (syncResponse && callback) {
|
|
callback(null, syncResponse);
|
|
}
|
|
else
|
|
{
|
|
var messageId = callbackId++;
|
|
|
|
if (Array.isArray(payload))
|
|
{
|
|
callbacks[messageId] = {num: payload.length,
|
|
results: [],
|
|
callback: callback};
|
|
for (var i in payload) {
|
|
bridgeSend({type: 'web3-send-async-read-only',
|
|
messageId: messageId,
|
|
payload: payload[i]});
|
|
}
|
|
}
|
|
else
|
|
{
|
|
callbacks[messageId] = {callback: callback};
|
|
bridgeSend({type: 'web3-send-async-read-only',
|
|
messageId: messageId,
|
|
payload: payload});
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
window.ethereum = new EthereumProvider();
|
|
})();
|