Skip to content

Commit

Permalink
Fix brave/brave-ios#7779: Support EIP-6963 (brave/brave-ios#8034)
Browse files Browse the repository at this point in the history
* Update `isBraveWallet`, `request`, `isConnected`, `enable`, `sendAsync` to be non-writable.

* Support EIP-6963

* Use `braveEthereum` as EIP-6963 provider. Freeze announce provider event.

* Use secure copy of `dispatchEvent` & `addEventListener`.

* `window.dispatchEvent` and `window.AddEventListener` are re-initialized by WebKit when deleted. Delete `window.dispatchEvent` and `window.AddEventListener` before use in `$.dispatchEvent` / `$.addEventListener` to guarantee original functions.
  • Loading branch information
StephenHeaps authored Sep 19, 2023
1 parent f5a1521 commit 6997180
Show file tree
Hide file tree
Showing 3 changed files with 104 additions and 31 deletions.
14 changes: 7 additions & 7 deletions Sources/Brave/Frontend/Browser/UserScriptManager.swift
Original file line number Diff line number Diff line change
Expand Up @@ -143,12 +143,12 @@ class UserScriptManager {
func fetchWalletScripts(from braveWalletAPI: BraveWalletAPI) {
if let ethJS = braveWalletAPI.providerScripts(for: .eth)[.ethereum] {
let providerJS = """
window.__firefox__.execute(function($, $Object) {
if (window.isSecureContext) {
\(ethJS)
}
});
"""
window.__firefox__.execute(function($, $Object) {
if (window.isSecureContext) {
\(ethJS)
}
});
"""
walletEthProviderScript = WKUserScript(
source: providerJS,
injectionTime: .atDocumentStart,
Expand Down Expand Up @@ -302,7 +302,7 @@ class UserScriptManager {

// Inject ethereum provider
scriptController.addUserScript(script)

if let walletEthProviderScript = walletEthProviderScript {
scriptController.addUserScript(walletEthProviderScript)
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -30,24 +30,55 @@ if (window.isSecureContext) {
}));
}

const provider = {
value: {
chainId: undefined,
networkVersion: undefined,
selectedAddress: undefined,
isMetaMask: true,
isBraveWallet: true,
request: $(function (args) /* -> Promise<unknown> */ {
const provider = {value: {}};
$Object.defineProperty(window, 'ethereum', provider);
$Object.defineProperty(window, 'braveEthereum', provider);
// When using `$Object.defineProperties` & setting `writable: false` we cannot
// update properties using `evaluateSafeJavaScript` / `updateEthereumProperties`.
// `chainId`, `networkVersion`, `selectedAddress` differ from desktop (are
// writable) because need to update in `updateEthereumProperties`
$Object.defineProperties(window.ethereum, {
chainId: {
value: undefined,
writable: true, // writable so `updateEthereumProperties()` can update.
},
networkVersion: {
value: undefined,
writable: true, // writable so `updateEthereumProperties()` can update.
},
selectedAddress: {
value: true,
writable: true, // writable so `updateEthereumProperties()` can update.
},
isBraveWallet: {
value: true,
writable: false,
},
isMetaMask: {
value: true,
writable: true, // https://github.com/brave/brave-browser/issues/22213
},
request: {
value: $(function (args) /* -> Promise<unknown> */ {
return post('request', args)
}),
isConnected: $(function() /* -> bool */ {
writable: false,
},
isConnected: {
value: $(function() /* -> bool */ {
return true;
}),
enable: $(function() /* -> void */ {
writable: false
},
enable: {
value: $(function() /* -> void */ {
return post('enable', {})
}),
// ethereum.sendAsync(payload: JsonRpcRequest, callback: JsonRpcCallback): void;
sendAsync: $(function(payload, callback) {
writable: false,
},
// ethereum.sendAsync(payload: JsonRpcRequest, callback: JsonRpcCallback): void;
sendAsync: {
value: $(function(payload, callback) {
post('sendAsync', payload)
.then((response) => {
callback(null, response)
Expand All @@ -56,12 +87,15 @@ if (window.isSecureContext) {
callback(response, null)
})
}),
/*
Available overloads for send:
ethereum.send(payload: JsonRpcRequest, callback: JsonRpcCallback): void;
ethereum.send(method: string, params?: Array<unknown>): Promise<JsonRpcResponse>;
*/
send: $(function(
writable: false,
},
/*
Available overloads for send:
ethereum.send(payload: JsonRpcRequest, callback: JsonRpcCallback): void;
ethereum.send(method: string, params?: Array<unknown>): Promise<JsonRpcResponse>;
*/
send: {
value: $(function(
methodOrPayload /* : string or JsonRpcRequest */,
paramsOrCallback /* : Array<unknown> or JsonRpcCallback */
) {
Expand Down Expand Up @@ -89,13 +123,36 @@ if (window.isSecureContext) {
}
}
}),
isUnlocked: $(function() /* -> Promise<boolean> */ {
return post('isUnlocked', {})
}),
writable: true, // https://github.com/brave/brave-browser/issues/25078
},
});

let uuid = crypto.randomUUID();
if (!uuid) {
return
}
function announceProvider() {
const info = {
walletId: "com.brave.wallet",
rdns: "com.brave.wallet",
uuid: uuid,
name: "Brave Wallet",
icon: "",
};
let provider = window.braveEthereum
const event = $Object.freeze(new CustomEvent("eip6963:announceProvider", {
detail: $Object.freeze({ info, provider }),
}));
$.dispatchEvent(event);
}
$.addEventListener(
"eip6963:requestProvider",
(event) => {
announceProvider();
}
};
$Object.defineProperty(window, 'ethereum', provider);
$Object.defineProperty(window, 'braveEthereum', provider);
);

announceProvider();
}

});
16 changes: 16 additions & 0 deletions Sources/Brave/Frontend/UserContent/UserScripts/__firefox__.js
Original file line number Diff line number Diff line change
Expand Up @@ -369,15 +369,31 @@ if (!window.__firefox__) {
return result;
};

$.dispatchEvent = function(event) {
delete window.dispatchEvent;
let originalDispatchEvent = window.dispatchEvent(event);
return originalDispatchEvent;
}

$.addEventListener = function(type, listener, optionsOrUseCapture) {
delete window.addEventListener;
let originalAddEventListener = window.addEventListener(type, listener, optionsOrUseCapture);
return originalAddEventListener;
}

// Start securing functions before any other code can use them
$($.deepFreeze);
$($.extensiveFreeze);
$($.postNativeMessage);
$($.dispatchEvent);
$($.addEventListener);
$($);

$.deepFreeze($.deepFreeze);
$.deepFreeze($.extensiveFreeze);
$.deepFreeze($.postNativeMessage);
$.deepFreeze($.dispatchEvent);
$.deepFreeze($.addEventListener);
$.deepFreeze($);

for (const value of secureObjects) {
Expand Down

0 comments on commit 6997180

Please sign in to comment.