Refactor processing Safari requests on macOS

This commit is contained in:
Ivan Grachev 2022-02-12 21:30:05 +03:00
parent efdb05a39f
commit 1e93d8ffe1
2 changed files with 20 additions and 128 deletions

View File

@ -89,32 +89,28 @@ class Agent: NSObject {
} }
} }
func showApprove(transaction: Transaction, chain: EthereumChain, peerMeta: PeerMeta?, browser: Browser?, completion: @escaping (Transaction?) -> Void) { func showApprove(transaction: Transaction, chain: EthereumChain, peerMeta: PeerMeta?, completion: @escaping (Transaction?) -> Void) {
let windowController = Window.showNew() let windowController = Window.showNew()
let approveViewController = ApproveTransactionViewController.with(transaction: transaction, chain: chain, peerMeta: peerMeta) { [weak self] transaction in let approveViewController = ApproveTransactionViewController.with(transaction: transaction, chain: chain, peerMeta: peerMeta) { [weak self] transaction in
if transaction != nil { if transaction != nil {
self?.askAuthentication(on: windowController.window, onStart: false, reason: .sendTransaction) { success in self?.askAuthentication(on: windowController.window, onStart: false, reason: .sendTransaction) { success in
completion(success ? transaction : nil) completion(success ? transaction : nil)
Window.closeAllAndActivateBrowser(force: browser)
} }
} else { } else {
Window.closeAllAndActivateBrowser(force: browser)
completion(nil) completion(nil)
} }
} }
windowController.contentViewController = approveViewController windowController.contentViewController = approveViewController
} }
func showApprove(subject: ApprovalSubject, meta: String, peerMeta: PeerMeta?, browser: Browser?, completion: @escaping (Bool) -> Void) { func showApprove(subject: ApprovalSubject, meta: String, peerMeta: PeerMeta?, completion: @escaping (Bool) -> Void) {
let windowController = Window.showNew() let windowController = Window.showNew()
let approveViewController = ApproveViewController.with(subject: subject, meta: meta, peerMeta: peerMeta) { [weak self] result in let approveViewController = ApproveViewController.with(subject: subject, meta: meta, peerMeta: peerMeta) { [weak self] result in
if result { if result {
self?.askAuthentication(on: windowController.window, onStart: false, reason: subject.asAuthenticationReason) { success in self?.askAuthentication(on: windowController.window, onStart: false, reason: subject.asAuthenticationReason) { success in
completion(success) completion(success)
Window.closeAllAndActivateBrowser(force: browser)
} }
} else { } else {
Window.closeAllAndActivateBrowser(force: browser)
completion(result) completion(result)
} }
} }
@ -301,130 +297,22 @@ class Agent: NSObject {
} }
private func processSafariRequest(_ safariRequest: SafariRequest) { private func processSafariRequest(_ safariRequest: SafariRequest) {
// TODO: process all chains' requests let action = DappRequestProcessor.processSafariRequest(safariRequest) {
guard ExtensionBridge.hasRequest(id: safariRequest.id), case let .ethereum(request) = safariRequest.body else { Window.closeAllAndActivateBrowser(force: .safari)
respond(to: safariRequest, error: Strings.somethingWentWrong)
return
} }
let peerMeta = PeerMeta(title: safariRequest.host, iconURLString: safariRequest.favicon) switch action {
switch request.method { case .none:
case .signPersonalMessage: break
guard let data = request.message else { case .selectAccount(let action):
respond(to: safariRequest, error: Strings.somethingWentWrong)
return
}
let text = String(data: data, encoding: .utf8) ?? data.hexString
showApprove(subject: .signPersonalMessage, meta: text, peerMeta: peerMeta, browser: .safari) { [weak self] approved in
if approved {
self?.signPersonalMessage(address: request.address, data: data, request: safariRequest)
} else {
self?.respond(to: safariRequest, error: Strings.failedToSign)
}
}
case .requestAccounts, .switchAccount:
let windowController = Window.showNew() let windowController = Window.showNew()
let accountsList = instantiate(AccountsListViewController.self) let accountsList = instantiate(AccountsListViewController.self)
accountsList.onSelectedWallet = action.completion
accountsList.onSelectedWallet = { [weak self] chain, wallet in
if let chain = chain, let wallet = wallet, let ethereumAddress = wallet.ethereumAddress {
let responseBody = ResponseToExtension.Ethereum(results: [ethereumAddress], chainId: chain.hexStringId, rpcURL: chain.nodeURLString)
self?.respond(to: safariRequest, body: .ethereum(responseBody))
} else {
self?.respond(to: safariRequest, error: Strings.canceled)
}
Window.closeAllAndActivateBrowser(force: .safari)
}
windowController.contentViewController = accountsList windowController.contentViewController = accountsList
case .signMessage: case .approveMessage(let action):
guard let data = request.message else { showApprove(subject: action.subject, meta: action.meta, peerMeta: action.peerMeta, completion: action.completion)
respond(to: safariRequest, error: Strings.somethingWentWrong) case .approveTransaction(let action):
return showApprove(transaction: action.transaction, chain: action.chain, peerMeta: action.peerMeta, completion: action.completion)
}
showApprove(subject: .signMessage, meta: data.hexString, peerMeta: peerMeta, browser: .safari) { [weak self] approved in
if approved {
self?.signMessage(address: request.address, data: data, request: safariRequest)
} else {
self?.respond(to: safariRequest, error: Strings.failedToSign)
}
}
case .signTypedMessage:
guard let raw = request.raw else {
respond(to: safariRequest, error: Strings.somethingWentWrong)
return
}
showApprove(subject: .signTypedData, meta: raw, peerMeta: peerMeta, browser: .safari) { [weak self] approved in
if approved {
self?.signTypedData(address: request.address, raw: raw, request: safariRequest)
} else {
self?.respond(to: safariRequest, error: Strings.failedToSign)
}
}
case .signTransaction:
guard let transaction = request.transaction, let chain = request.chain else {
respond(to: safariRequest, error: Strings.somethingWentWrong)
return
}
showApprove(transaction: transaction, chain: chain, peerMeta: peerMeta, browser: .safari) { [weak self] transaction in
if let transaction = transaction {
self?.sendTransaction(transaction, address: request.address, chain: chain, request: safariRequest)
} else {
self?.respond(to: safariRequest, error: Strings.canceled)
}
}
case .ecRecover:
if let (signature, message) = request.signatureAndMessage,
let recovered = ethereum.recover(signature: signature, message: message) {
respond(to: safariRequest, body: .ethereum(.init(result: recovered)))
} else {
respond(to: safariRequest, error: Strings.failedToVerify)
}
Window.closeAllAndActivateBrowser(force: .safari)
case .switchEthereumChain, .addEthereumChain, .watchAsset:
Window.closeAllAndActivateBrowser(force: .safari)
}
}
private func respond(to safariRequest: SafariRequest, body: ResponseToExtension.Body) {
let response = ResponseToExtension(for: safariRequest, body: body)
ExtensionBridge.respond(response: response)
}
private func respond(to safariRequest: SafariRequest, error: String) {
let response = ResponseToExtension(for: safariRequest, error: error)
ExtensionBridge.respond(response: response)
}
private func sendTransaction(_ transaction: Transaction, address: String, chain: EthereumChain, request: SafariRequest) {
if let wallet = walletsManager.getWallet(address: address),
let transactionHash = try? ethereum.send(transaction: transaction, wallet: wallet, chain: chain) {
respond(to: request, body: .ethereum(.init(result: transactionHash)))
} else {
respond(to: request, error: Strings.failedToSign)
}
}
private func signTypedData(address: String, raw: String, request: SafariRequest) {
if let wallet = walletsManager.getWallet(address: address), let signed = try? ethereum.sign(typedData: raw, wallet: wallet) {
respond(to: request, body: .ethereum(.init(result: signed)))
} else {
respond(to: request, error: Strings.failedToSign)
}
}
private func signMessage(address: String, data: Data, request: SafariRequest) {
if let wallet = walletsManager.getWallet(address: address), let signed = try? ethereum.sign(data: data, wallet: wallet) {
respond(to: request, body: .ethereum(.init(result: signed)))
} else {
respond(to: request, error: Strings.failedToSign)
}
}
private func signPersonalMessage(address: String, data: Data, request: SafariRequest) {
if let wallet = walletsManager.getWallet(address: address), let signed = try? ethereum.signPersonalMessage(data: data, wallet: wallet) {
respond(to: request, body: .ethereum(.init(result: signed)))
} else {
respond(to: request, error: Strings.failedToSign)
} }
} }

View File

@ -133,10 +133,12 @@ class WalletConnect {
let peer = PeerMeta(wcPeerMeta: getPeerOfInteractor(interactor)) let peer = PeerMeta(wcPeerMeta: getPeerOfInteractor(interactor))
let transaction = Transaction(from: wct.from, to: to, nonce: wct.nonce, gasPrice: wct.gasPrice, gas: wct.gas, value: wct.value, data: wct.data) let transaction = Transaction(from: wct.from, to: to, nonce: wct.nonce, gasPrice: wct.gasPrice, gas: wct.gas, value: wct.value, data: wct.data)
agent.showApprove(transaction: transaction, chain: chain, peerMeta: peer, browser: nil) { [weak self, weak interactor] transaction in agent.showApprove(transaction: transaction, chain: chain, peerMeta: peer) { [weak self, weak interactor] transaction in
if let transaction = transaction { if let transaction = transaction {
self?.sendTransaction(transaction, walletId: walletId, chainId: chainId, requestId: id, interactor: interactor) self?.sendTransaction(transaction, walletId: walletId, chainId: chainId, requestId: id, interactor: interactor)
Window.closeAllAndActivateBrowser(force: nil)
} else { } else {
Window.closeAllAndActivateBrowser(force: nil)
self?.rejectRequest(id: id, interactor: interactor, message: Strings.canceled) self?.rejectRequest(id: id, interactor: interactor, message: Strings.canceled)
} }
} }
@ -160,10 +162,12 @@ class WalletConnect {
} }
let peer = PeerMeta(wcPeerMeta: getPeerOfInteractor(interactor)) let peer = PeerMeta(wcPeerMeta: getPeerOfInteractor(interactor))
agent.showApprove(subject: approvalSubject, meta: message ?? "", peerMeta: peer, browser: nil) { [weak self, weak interactor] approved in agent.showApprove(subject: approvalSubject, meta: message ?? "", peerMeta: peer) { [weak self, weak interactor] approved in
if approved { if approved {
self?.sign(id: id, payload: payload, walletId: walletId, interactor: interactor) self?.sign(id: id, payload: payload, walletId: walletId, interactor: interactor)
Window.closeAllAndActivateBrowser(force: nil)
} else { } else {
Window.closeAllAndActivateBrowser(force: nil)
self?.rejectRequest(id: id, interactor: interactor, message: Strings.canceled) self?.rejectRequest(id: id, interactor: interactor, message: Strings.canceled)
} }
} }