2024-04-30 14:08:37 +02:00
import QtQuick 2.15
2024-05-31 12:58:47 +03:00
2024-04-30 14:08:37 +02:00
import QtTest 1.15
2024-05-31 12:58:47 +03:00
import "helpers/wallet_connect.js" as Testing
2024-04-30 14:08:37 +02:00
2024-04-30 15:21:19 +02:00
import StatusQ 0.1 // See #10218
2024-05-31 12:58:47 +03:00
import StatusQ . Core . Utils 0.1
2024-04-30 15:21:19 +02:00
2024-04-30 14:08:37 +02:00
import QtQuick . Controls 2.15
import Storybook 1.0
2024-05-06 22:22:43 +02:00
import AppLayouts . Wallet . services . dapps 1.0
2024-05-31 12:58:47 +03:00
import AppLayouts . Wallet . services . dapps . types 1.0
2024-05-31 12:34:59 +03:00
import AppLayouts . Profile . stores 1.0
import AppLayouts . Wallet . panels 1.0
2024-07-17 18:49:30 +03:00
import AppLayouts . Wallet . stores 1.0 as WalletStore
2024-05-06 22:22:43 +02:00
2024-05-31 12:58:47 +03:00
import shared . stores 1.0
import utils 1.0
2024-04-30 14:08:37 +02:00
Item {
id: root
2024-05-31 12:58:47 +03:00
2024-06-04 23:45:03 +03:00
width: 600
height: 400
2024-05-31 12:58:47 +03:00
2024-07-04 16:32:04 +03:00
Component {
id: sdkComponent
WalletConnectSDKBase {
property bool sdkReady: true
property var getActiveSessionsCallbacks: [ ]
getActiveSessions: function ( callback ) {
getActiveSessionsCallbacks . push ( { callback } )
}
property int pairCalled: 0
pair: function ( ) {
pairCalled ++
}
property var buildApprovedNamespacesCalls: [ ]
buildApprovedNamespaces: function ( params , supportedNamespaces ) {
buildApprovedNamespacesCalls . push ( { params , supportedNamespaces } )
}
property var approveSessionCalls: [ ]
approveSession: function ( sessionProposalJson , approvedNamespaces ) {
approveSessionCalls . push ( { sessionProposalJson , approvedNamespaces } )
}
2024-07-15 20:30:27 +03:00
property var acceptSessionRequestCalls: [ ]
acceptSessionRequest: function ( topic , id , signature ) {
acceptSessionRequestCalls . push ( { topic , id , signature } )
}
2024-07-04 16:32:04 +03:00
property var rejectSessionRequestCalls: [ ]
rejectSessionRequest: function ( topic , id , error ) {
rejectSessionRequestCalls . push ( { topic , id , error } )
}
}
}
Component {
id: serviceComponent
WalletConnectService {
property var onApproveSessionResultTriggers: [ ]
onApproveSessionResult: function ( session , error ) {
onApproveSessionResultTriggers . push ( { session , error } )
}
property var onDisplayToastMessageTriggers: [ ]
onDisplayToastMessage: function ( message , error ) {
onDisplayToastMessageTriggers . push ( { message , error } )
}
2024-07-24 15:50:54 +03:00
property var onPairingValidatedTriggers: [ ]
onPairingValidated: function ( validationState ) {
onPairingValidatedTriggers . push ( { validationState } )
}
2024-07-04 16:32:04 +03:00
}
}
Component {
id: dappsStoreComponent
DAppsStore {
property string dappsListReceivedJsonStr: '[]'
signal dappsListReceived ( string dappsJson )
signal userAuthenticated ( string topic , string id , string password , string pin )
signal userAuthenticationFailed ( string topic , string id )
// By default, return no dapps in store
function getDapps ( ) {
dappsListReceived ( dappsListReceivedJsonStr )
return true
}
property var addWalletConnectSessionCalls: [ ]
function addWalletConnectSession ( sessionJson ) {
addWalletConnectSessionCalls . push ( { sessionJson } )
2024-07-24 15:50:54 +03:00
return true
2024-07-04 16:32:04 +03:00
}
property var authenticateUserCalls: [ ]
function authenticateUser ( topic , id , address ) {
authenticateUserCalls . push ( { topic , id , address } )
}
property var signMessageCalls: [ ]
function signMessage ( topic , id , address , password , message ) {
signMessageCalls . push ( { topic , id , address , password , message } )
}
2024-07-02 00:02:05 +03:00
property var safeSignTypedDataCalls: [ ]
function safeSignTypedData ( topic , id , address , password , message , chainId , legacy ) {
safeSignTypedDataCalls . push ( { topic , id , address , password , message , chainId , legacy } )
2024-07-04 16:32:04 +03:00
}
2024-06-27 17:15:17 +03:00
property var updateWalletConnectSessionsCalls: [ ]
function updateWalletConnectSessions ( activeTopicsJson ) {
updateWalletConnectSessionsCalls . push ( { activeTopicsJson } )
}
2024-07-17 18:49:30 +03:00
function getEstimatedTime ( chainId , maxFeePerGas ) {
return Constants . TransactionEstimatedTime . LessThanThreeMins
}
property var mockedSuggestedFees: ( {
gasPrice: 2.0 ,
baseFee: 5.0 ,
maxPriorityFeePerGas: 2.0 ,
maxFeePerGasL: 1.0 ,
maxFeePerGasM: 1.1 ,
maxFeePerGasH: 1.2 ,
l1GasFee: 0.0 ,
eip1559Enabled: true
} )
2024-07-17 19:36:32 +03:00
2024-07-17 18:49:30 +03:00
function getSuggestedFees ( ) {
return mockedSuggestedFees
}
2024-07-17 19:36:32 +03:00
function hexToDec ( hex ) {
if ( hex . length > "0xfffffffffffff" . length ) {
console . warn ( ` Beware of possible loss of precision converting $ { hex } ` )
}
return parseInt ( hex , 16 ) . toString ( )
}
2024-07-04 16:32:04 +03:00
}
}
Component {
id: walletStoreComponent
2024-07-03 22:46:00 +02:00
QtObject {
2024-06-29 23:24:05 +03:00
readonly property ListModel filteredFlatModel: ListModel {
2024-07-17 18:49:30 +03:00
ListElement {
chainId: 1
layer: 1
}
2024-07-04 16:32:04 +03:00
ListElement {
chainId: 2
chainName: "Test Chain"
iconUrl: "network/Network=Ethereum"
2024-07-17 18:49:30 +03:00
layer: 2
}
// Used by tst_balanceCheck
ListElement {
chainId: 11155111
layer: 1
}
// Used by tst_balanceCheck
ListElement {
chainId: 421613
layer: 2
2024-07-04 16:32:04 +03:00
}
}
2024-06-29 23:24:05 +03:00
readonly property ListModel nonWatchAccounts: ListModel {
2024-07-04 16:32:04 +03:00
ListElement { address: "0x1" }
ListElement {
address: "0x2"
name: "helloworld"
emoji: "😋"
color: "#2A4AF5"
}
ListElement { address: "0x3a" }
2024-07-17 18:49:30 +03:00
// Account from GroupedAccountsAssetsModel
ListElement { address: "0x7F47C2e18a4BBf5487E6fb082eC2D9Ab0E6d7240" }
2024-07-04 16:32:04 +03:00
}
2024-06-29 23:24:05 +03:00
function getNetworkShortNames ( chainIds ) {
return "eth:oeth:arb"
2024-07-17 18:49:30 +03:00
}
readonly property var currencyStore: CurrenciesStore { }
readonly property var walletAssetsStore: assetsStoreMock
2024-07-04 16:32:04 +03:00
}
}
2024-07-17 18:49:30 +03:00
WalletStore . WalletAssetsStore {
id: assetsStoreMock
// Silence warnings
assetsWithFilteredBalances: ListModel { }
readonly property var groupedAccountAssetsModel: groupedAccountsAssetsModel
}
2024-07-04 16:32:04 +03:00
Component {
id: dappsRequestHandlerComponent
DAppsRequestHandler {
2024-07-17 13:46:43 +03:00
currenciesStore: CurrenciesStore { }
2024-07-17 18:49:30 +03:00
assetsStore: assetsStoreMock
property var maxFeesUpdatedCalls: [ ]
onMaxFeesUpdated: function ( fiatMaxFees , ethMaxFees , haveEnoughFunds , haveEnoughForFees , symbol , feesInfo ) {
maxFeesUpdatedCalls . push ( { fiatMaxFees , ethMaxFees , haveEnoughFunds , haveEnoughForFees , symbol , feesInfo } )
}
2024-07-04 16:32:04 +03:00
}
}
TestCase {
id: requestHandlerTest
name: "DAppsRequestHandler"
2024-07-17 18:49:30 +03:00
// Ensure mocked GroupedAccountsAssetsModel is properly initialized
when: windowShown
2024-07-04 16:32:04 +03:00
property DAppsRequestHandler handler: null
SignalSpy {
id: displayToastMessageSpy
target: requestHandlerTest . handler
signalName: "onDisplayToastMessage"
}
function init ( ) {
let walletStore = createTemporaryObject ( walletStoreComponent , root )
verify ( ! ! walletStore )
let sdk = createTemporaryObject ( sdkComponent , root , { projectId: "12ab" } )
verify ( ! ! sdk )
let store = createTemporaryObject ( dappsStoreComponent , root )
verify ( ! ! store )
2024-06-29 23:24:05 +03:00
handler = createTemporaryObject ( dappsRequestHandlerComponent , root , {
sdk: sdk ,
store: store ,
accountsModel: walletStore . nonWatchAccounts ,
networksModel: walletStore . filteredFlatModel
} )
2024-07-04 16:32:04 +03:00
verify ( ! ! handler )
}
function cleanup ( ) {
displayToastMessageSpy . clear ( )
}
function test_TestAuthentication ( ) {
2024-06-29 23:24:05 +03:00
let td = mockSessionRequestEvent ( this , handler . sdk , handler . accountsModel , handler . networksModel )
2024-07-04 16:32:04 +03:00
handler . authenticate ( td . request )
compare ( handler . store . authenticateUserCalls . length , 1 , "expected a call to store.authenticateUser" )
let store = handler . store
store . userAuthenticated ( td . topic , td . request . id , "password" , "" )
compare ( store . signMessageCalls . length , 1 , "expected a call to store.signMessage" )
compare ( store . signMessageCalls [ 0 ] . message , td . request . data )
}
function test_onSessionRequestEventDifferentCaseForAddress ( ) {
2024-07-29 19:46:03 +03:00
const sdk = handler . sdk
const testAddressUpper = "0x3A"
const chainId = 2
const method = "personal_sign"
const message = "hello world"
const params = [ ` "${DAppsHelpers.strToHex(message)}" ` , ` "${testAddressUpper}" ` ]
const topic = "b536a"
const session = JSON . parse ( Testing . formatSessionRequest ( chainId , method , params , topic ) )
2024-07-04 16:32:04 +03:00
// Expect to have calls to getActiveSessions from service initialization
2024-07-29 19:46:03 +03:00
const prevRequests = sdk . getActiveSessionsCallbacks . length
2024-07-04 16:32:04 +03:00
sdk . sessionRequestEvent ( session )
compare ( sdk . getActiveSessionsCallbacks . length , 1 , "expected DAppsRequestHandler call sdk.getActiveSessions" )
}
2024-07-17 18:49:30 +03:00
2024-07-23 13:49:34 +03:00
// Tests that the request is ignored if not in the current profile (don't have the PK for the address)
function test_onSessionRequestEventMissingAddress ( ) {
2024-07-29 19:46:03 +03:00
const sdk = handler . sdk
const testAddressUpper = "0xY"
const chainId = 2
const method = "personal_sign"
const message = "hello world"
const params = [ ` "${DAppsHelpers.strToHex(message)}" ` , ` "${testAddressUpper}" ` ]
const topic = "b536a"
const session = JSON . parse ( Testing . formatSessionRequest ( chainId , method , params , topic ) )
2024-07-23 13:49:34 +03:00
// Expect to have calls to getActiveSessions from service initialization
2024-07-29 19:46:03 +03:00
const prevRequests = sdk . getActiveSessionsCallbacks . length
2024-07-23 13:49:34 +03:00
sdk . sessionRequestEvent ( session )
compare ( sdk . getActiveSessionsCallbacks . length , 0 , "expected DAppsRequestHandler don't call sdk.getActiveSessions" )
compare ( sdk . rejectSessionRequestCalls . length , 0 , "expected no call to service.rejectSessionRequest" )
}
2024-07-17 18:49:30 +03:00
function test_balanceCheck_data ( ) {
return [ {
tag: "have_enough_funds" ,
chainId: 11155111 ,
expect: {
haveEnoughForFees: true ,
}
} ,
{
tag: "doest_have_enough_funds" ,
chainId: 11155111 ,
// Override the suggestedFees to a higher value
maxFeePerGasM: 1000000.0 , /*GWEI*/
expect: {
haveEnoughForFees: false ,
}
} ,
{
tag: "check_l2_doesnt_have_enough_funds_on_l1" ,
chainId: 421613 ,
// Override the l1 additional fees
l1GasFee: 1000000000.0 ,
expect: {
haveEnoughForFees: false ,
}
} ,
{
tag: "check_l2_doesnt_have_enough_funds_on_l2" ,
chainId: 421613 ,
// Override the l2 to a higher value
maxFeePerGasM: 1000000.0 , /*GWEI*/
// Override the l1 additional fees
l1GasFee: 10.0 ,
expect: {
haveEnoughForFees: false ,
}
} ]
}
function test_balanceCheck ( data ) {
let sdk = handler . sdk
// Override the suggestedFees
if ( ! ! data . maxFeePerGasM ) {
handler . store . mockedSuggestedFees . maxFeePerGasM = data . maxFeePerGasM
}
if ( ! ! data . l1GasFee ) {
handler . store . mockedSuggestedFees . l1GasFee = data . l1GasFee
}
let testAddress = "0x7F47C2e18a4BBf5487E6fb082eC2D9Ab0E6d7240"
let chainId = data . chainId
let method = "eth_sendTransaction"
let message = "hello world"
let params = [ ` {
"data" : "0x" ,
"from" : "${testAddress}" ,
"to" : "0x2" ,
"value" : "0x12345"
} ` ]
let topic = "b536a"
let session = JSON . parse ( Testing . formatSessionRequest ( chainId , method , params , topic ) )
sdk . sessionRequestEvent ( session )
compare ( sdk . getActiveSessionsCallbacks . length , 1 , "expected DAppsRequestHandler call sdk.getActiveSessions" )
let callback = sdk . getActiveSessionsCallbacks [ 0 ] . callback
callback ( { "b536a" : JSON . parse ( Testing . formatApproveSessionResponse ( [ chainId , 7 ] , [ testAddress ] ) ) } )
compare ( handler . maxFeesUpdatedCalls . length , 1 , "expected a call to handler.onMaxFeesUpdated" )
let args = handler . maxFeesUpdatedCalls [ 0 ]
verify ( args . ethMaxFees > 0 , "expected ethMaxFees to be set" )
// storybook's CurrenciesStore mock up getFiatValue returns the balance
compare ( args . fiatMaxFees . toString ( ) , args . ethMaxFees . toString ( ) , "expected fiatMaxFees to be set" )
verify ( args . haveEnoughFunds , "expected haveEnoughFunds to be set" )
compare ( args . haveEnoughForFees , data . expect . haveEnoughForFees , "expected haveEnoughForFees to be set" )
verify ( ! ! args . feesInfo , "expected feesInfo to be set" )
}
2024-07-04 16:32:04 +03:00
}
TestCase {
id: walletConnectServiceTest
name: "WalletConnectService"
property WalletConnectService service: null
SignalSpy {
id: connectDAppSpy
target: walletConnectServiceTest . service
signalName: "connectDApp"
property var argPos: {
"dappChains" : 0 ,
"sessionProposalJson" : 1 ,
"availableNamespaces" : 0
}
}
2024-07-17 18:49:30 +03:00
readonly property SignalSpy sessionRequestSpy: SignalSpy {
2024-07-04 16:32:04 +03:00
target: walletConnectServiceTest . service
signalName: "sessionRequest"
property var argPos: {
"request" : 0
}
}
function init ( ) {
let walletStore = createTemporaryObject ( walletStoreComponent , root )
verify ( ! ! walletStore )
let sdk = createTemporaryObject ( sdkComponent , root , { projectId: "12ab" } )
verify ( ! ! sdk )
let store = createTemporaryObject ( dappsStoreComponent , root )
verify ( ! ! store )
2024-06-29 23:24:05 +03:00
service = createTemporaryObject ( serviceComponent , root , { wcSDK: sdk , store: store , walletRootStore: walletStore } )
2024-07-04 16:32:04 +03:00
verify ( ! ! service )
}
function cleanup ( ) {
connectDAppSpy . clear ( )
sessionRequestSpy . clear ( )
}
2024-07-24 15:50:54 +03:00
function testSetupPair ( sessionProposalPayload ) {
2024-07-04 16:32:04 +03:00
let sdk = service . wcSDK
2024-06-29 23:24:05 +03:00
let walletStore = service . walletRootStore
2024-07-04 16:32:04 +03:00
let store = service . store
service . pair ( "wc:12ab@1?bridge=https%3A%2F%2Fbridge.walletconnect.org&key=12ab" )
compare ( sdk . pairCalled , 1 , "expected a call to sdk.pair" )
2024-07-24 15:50:54 +03:00
sdk . sessionProposal ( JSON . parse ( sessionProposalPayload ) )
2024-07-04 16:32:04 +03:00
compare ( sdk . buildApprovedNamespacesCalls . length , 1 , "expected a call to sdk.buildApprovedNamespaces" )
var args = sdk . buildApprovedNamespacesCalls [ 0 ]
verify ( ! ! args . supportedNamespaces , "expected supportedNamespaces to be set" )
2024-07-24 15:50:54 +03:00
// All calls to SDK are expected as events to be made by the wallet connect SDK
2024-07-04 16:32:04 +03:00
let chainsForApproval = args . supportedNamespaces . eip155 . chains
2024-06-29 23:24:05 +03:00
let networksArray = ModelUtils . modelToArray ( walletStore . filteredFlatModel ) . map ( entry = > entry . chainId )
2024-07-04 16:32:04 +03:00
verify ( networksArray . every ( chainId = > chainsForApproval . some ( eip155Chain = > eip155Chain === ` eip155: $ { chainId } ` ) ) ,
"expect all the networks to be present" )
// We test here all accounts for one chain only, we have separate tests to validate that all accounts are present
let allAccountsForApproval = args . supportedNamespaces . eip155 . accounts
2024-06-29 23:24:05 +03:00
let accountsArray = ModelUtils . modelToArray ( walletStore . nonWatchAccounts ) . map ( entry = > entry . address )
2024-07-04 16:32:04 +03:00
verify ( accountsArray . every ( address = > allAccountsForApproval . some ( eip155Address = > eip155Address === ` eip155: $ { networksArray [ 0 ] } : $ { address } ` ) ) ,
"expect at least all accounts for the first chain to be present"
)
2024-07-24 15:50:54 +03:00
return { sdk , walletStore , store , networksArray , accountsArray }
}
function test_TestPairing ( ) {
const { sdk , walletStore , store , networksArray , accountsArray } = testSetupPair ( Testing . formatSessionProposal ( ) )
2024-07-04 16:32:04 +03:00
let allApprovedNamespaces = JSON . parse ( Testing . formatBuildApprovedNamespacesResult ( networksArray , accountsArray ) )
sdk . buildApprovedNamespacesResult ( allApprovedNamespaces , "" )
compare ( connectDAppSpy . count , 1 , "expected a call to service.connectDApp" )
let connectArgs = connectDAppSpy . signalArguments [ 0 ]
2024-06-29 23:24:05 +03:00
compare ( connectArgs [ connectDAppSpy . argPos . dappChains ] , networksArray , "expected all provided networks (walletStore.filteredFlatModel) for the dappChains" )
2024-07-04 16:32:04 +03:00
verify ( ! ! connectArgs [ connectDAppSpy . argPos . sessionProposalJson ] , "expected sessionProposalJson to be set" )
verify ( ! ! connectArgs [ connectDAppSpy . argPos . availableNamespaces ] , "expected availableNamespaces to be set" )
2024-06-29 23:24:05 +03:00
let selectedAccount = walletStore . nonWatchAccounts . get ( 1 )
2024-07-04 16:32:04 +03:00
service . approvePairSession ( connectArgs [ connectDAppSpy . argPos . sessionProposalJson ] , connectArgs [ connectDAppSpy . argPos . dappChains ] , selectedAccount )
compare ( sdk . buildApprovedNamespacesCalls . length , 2 , "expected a call to sdk.buildApprovedNamespaces" )
2024-07-24 15:50:54 +03:00
const approvedArgs = sdk . buildApprovedNamespacesCalls [ 1 ]
verify ( ! ! approvedArgs . supportedNamespaces , "expected supportedNamespaces to be set" )
2024-07-04 16:32:04 +03:00
// We test here that only one account for all chains is provided
2024-07-24 15:50:54 +03:00
let accountsForApproval = approvedArgs . supportedNamespaces . eip155 . accounts
2024-07-04 16:32:04 +03:00
compare ( accountsForApproval . length , networksArray . length , "expect only one account per chain" )
compare ( accountsForApproval [ 0 ] , ` eip155: $ { networksArray [ 0 ] } : $ { selectedAccount . address } ` )
compare ( accountsForApproval [ 1 ] , ` eip155: $ { networksArray [ 1 ] } : $ { selectedAccount . address } ` )
let approvedNamespaces = JSON . parse ( Testing . formatBuildApprovedNamespacesResult ( networksArray , [ selectedAccount . address ] ) )
sdk . buildApprovedNamespacesResult ( approvedNamespaces , "" )
compare ( sdk . approveSessionCalls . length , 1 , "expected a call to sdk.approveSession" )
verify ( ! ! sdk . approveSessionCalls [ 0 ] . sessionProposalJson , "expected sessionProposalJson to be set" )
verify ( ! ! sdk . approveSessionCalls [ 0 ] . approvedNamespaces , "expected approvedNamespaces to be set" )
let finalApprovedNamespaces = JSON . parse ( Testing . formatApproveSessionResponse ( networksArray , [ selectedAccount . address ] ) )
sdk . approveSessionResult ( finalApprovedNamespaces , "" )
verify ( store . addWalletConnectSessionCalls . length === 1 )
verify ( store . addWalletConnectSessionCalls [ 0 ] . sessionJson , "expected sessionJson to be set" )
verify ( service . onApproveSessionResultTriggers . length === 1 )
verify ( service . onApproveSessionResultTriggers [ 0 ] . session , "expected session to be set" )
compare ( service . onDisplayToastMessageTriggers . length , 1 , "expected a success message to be displayed" )
verify ( ! service . onDisplayToastMessageTriggers [ 0 ] . error , "expected no error" )
verify ( service . onDisplayToastMessageTriggers [ 0 ] . message , "expected message to be set" )
}
2024-07-24 15:50:54 +03:00
function test_TestPairingUnsupportedNetworks ( ) {
const { sdk , walletStore , store } = testSetupPair ( Testing . formatSessionProposal ( ) )
let allApprovedNamespaces = JSON . parse ( Testing . formatBuildApprovedNamespacesResult ( [ ] , [ ] ) )
sdk . buildApprovedNamespacesResult ( allApprovedNamespaces , "" )
compare ( connectDAppSpy . count , 0 , "expected not to have calls to service.connectDApp" )
compare ( service . onPairingValidatedTriggers . length , 1 , "expected a call to service.onPairingValidated" )
compare ( service . onPairingValidatedTriggers [ 0 ] . validationState , Pairing . errors . unsupportedNetwork , "expected unsupportedNetwork state error" )
}
2024-07-04 16:32:04 +03:00
function test_SessionRequestMainFlow ( ) {
// All calls to SDK are expected as events to be made by the wallet connect SDK
2024-07-29 19:46:03 +03:00
const sdk = service . wcSDK
const walletStore = service . walletRootStore
const store = service . store
const testAddress = "0x3a"
const chainId = 2
const method = "personal_sign"
const message = "hello world"
const params = [ ` "${DAppsHelpers.strToHex(message)}" ` , ` "${testAddress}" ` ]
const topic = "b536a"
const session = JSON . parse ( Testing . formatSessionRequest ( chainId , method , params , topic ) )
2024-07-04 16:32:04 +03:00
// Expect to have calls to getActiveSessions from service initialization
2024-07-29 19:46:03 +03:00
const prevRequests = sdk . getActiveSessionsCallbacks . length
2024-07-04 16:32:04 +03:00
sdk . sessionRequestEvent ( session )
compare ( sdk . getActiveSessionsCallbacks . length , prevRequests + 1 , "expected DAppsRequestHandler call sdk.getActiveSessions" )
2024-07-29 19:46:03 +03:00
const callback = sdk . getActiveSessionsCallbacks [ prevRequests ] . callback
2024-07-04 16:32:04 +03:00
callback ( { "b536a" : JSON . parse ( Testing . formatApproveSessionResponse ( [ chainId , 7 ] , [ testAddress ] ) ) } )
compare ( sessionRequestSpy . count , 1 , "expected service.sessionRequest trigger" )
2024-07-29 19:46:03 +03:00
const request = sessionRequestSpy . signalArguments [ 0 ] [ sessionRequestSpy . argPos . request ]
2024-07-04 16:32:04 +03:00
compare ( request . topic , topic , "expected topic to be set" )
compare ( request . method , method , "expected method to be set" )
compare ( request . event , session , "expected event to be the one sent by the sdk" )
compare ( request . dappName , Testing . dappName , "expected dappName to be set" )
compare ( request . dappUrl , Testing . dappUrl , "expected dappUrl to be set" )
compare ( request . dappIcon , Testing . dappFirstIcon , "expected dappIcon to be set" )
verify ( ! ! request . account , "expected account to be set" )
compare ( request . account . address , testAddress , "expected look up of the right account" )
verify ( ! ! request . network , "expected network to be set" )
compare ( request . network . chainId , chainId , "expected look up of the right network" )
verify ( ! ! request . data , "expected data to be set" )
compare ( request . data . message , message , "expected message to be set" )
}
2024-07-24 15:50:54 +03:00
2024-07-04 16:32:04 +03:00
// TODO #14757: add tests with multiple session requests coming in; validate that authentication is serialized and in order
// function tst_SessionRequestQueueMultiple() {
// }
}
Component {
id: dappsListProviderComponent
DAppsListProvider {
}
}
TestCase {
name: "DAppsListProvider"
property DAppsListProvider provider: null
readonly property var dappsListReceivedJsonStr: '[{"url":"https://tst1.com","name":"name1","iconUrl":"https://tst1.com/u/1"},{"url":"https://tst2.com","name":"name2","iconUrl":"https://tst2.com/u/2"}]'
function init ( ) {
// Simulate the SDK not being ready
let sdk = createTemporaryObject ( sdkComponent , root , { projectId: "12ab" , sdkReady: false } )
verify ( ! ! sdk )
let store = createTemporaryObject ( dappsStoreComponent , root , {
dappsListReceivedJsonStr: dappsListReceivedJsonStr
} )
verify ( ! ! store )
2024-07-23 17:26:55 +03:00
const walletStore = createTemporaryObject ( walletStoreComponent , root )
verify ( ! ! walletStore )
provider = createTemporaryObject ( dappsListProviderComponent , root , { sdk: sdk , store: store , supportedAccountsModel: walletStore . nonWatchAccounts } )
2024-07-04 16:32:04 +03:00
verify ( ! ! provider )
}
function cleanup ( ) {
}
// Implemented as a regression to metamask not having icons which failed dapps list
function test_TestUpdateDapps ( ) {
provider . updateDapps ( )
// Validate that persistance fallback is working
compare ( provider . dappsModel . count , 2 , "expected dappsModel have the right number of elements" )
let persistanceList = JSON . parse ( dappsListReceivedJsonStr )
compare ( provider . dappsModel . get ( 0 ) . url , persistanceList [ 0 ] . url , "expected url to be set" )
compare ( provider . dappsModel . get ( 0 ) . iconUrl , persistanceList [ 0 ] . iconUrl , "expected iconUrl to be set" )
compare ( provider . dappsModel . get ( 1 ) . name , persistanceList [ 1 ] . name , "expected name to be set" )
// Validate that SDK's `getActiveSessions` is not called if not ready
let sdk = provider . sdk
compare ( sdk . getActiveSessionsCallbacks . length , 0 , "expected no calls to sdk.getActiveSessions yet" )
sdk . sdkReady = true
compare ( sdk . getActiveSessionsCallbacks . length , 1 , "expected a call to sdk.getActiveSessions when SDK becomes ready" )
let callback = sdk . getActiveSessionsCallbacks [ 0 ] . callback
2024-07-23 17:26:55 +03:00
const address = ModelUtils . get ( provider . supportedAccountsModel , 0 , "address" )
let session = JSON . parse ( Testing . formatApproveSessionResponse ( [ 1 , 2 ] , [ address ] , { dappMetadataJsonString: Testing . noIconsDappMetadataJsonString } ) )
2024-07-04 16:32:04 +03:00
callback ( { "b536a" : session , "b537b" : session } )
compare ( provider . dappsModel . count , 1 , "expected dappsModel have the SDK's reported dapps" )
compare ( provider . dappsModel . get ( 0 ) . iconUrl , "" , "expected iconUrl to be missing" )
2024-06-27 17:15:17 +03:00
let updateCalls = provider . store . updateWalletConnectSessionsCalls
compare ( updateCalls . length , 1 , "expected a call to store.updateWalletConnectSessions" )
verify ( updateCalls [ 0 ] . activeTopicsJson . includes ( "b536a" ) )
verify ( updateCalls [ 0 ] . activeTopicsJson . includes ( "b537b" ) )
2024-07-04 16:32:04 +03:00
}
}
TestCase {
name: "ServiceHelpers"
function test_extractChainsAndAccountsFromApprovedNamespaces ( ) {
2024-07-29 19:46:03 +03:00
const res = DAppsHelpers . extractChainsAndAccountsFromApprovedNamespaces ( JSON . parse ( ` {
2024-07-04 16:32:04 +03:00
"eip155" : {
"accounts" : [
"eip155:1:0x1" ,
"eip155:1:0x2" ,
"eip155:2:0x1" ,
"eip155:2:0x2"
] ,
"chains" : [
"eip155:1" ,
"eip155:2"
] ,
"events" : [
"accountsChanged" ,
"chainChanged"
] ,
"methods" : [
"eth_sendTransaction" ,
"personal_sign"
]
}
} ` ) )
verify ( res . chains . length === 2 )
verify ( res . accounts . length === 2 )
verify ( res . chains [ 0 ] === 1 )
verify ( res . chains [ 1 ] === 2 )
verify ( res . accounts [ 0 ] === "0x1" )
verify ( res . accounts [ 1 ] === "0x2" )
}
readonly property ListModel chainsModel: ListModel {
ListElement { chainId: 1 }
ListElement { chainId: 2 }
}
readonly property ListModel accountsModel: ListModel {
ListElement { address: "0x1" }
ListElement { address: "0x2" }
}
function test_buildSupportedNamespacesFromModels ( ) {
2024-07-29 19:46:03 +03:00
const methods = [ "eth_sendTransaction" , "personal_sign" ]
const resStr = DAppsHelpers . buildSupportedNamespacesFromModels ( chainsModel , accountsModel , methods )
const jsonObj = JSON . parse ( resStr )
2024-07-04 16:32:04 +03:00
verify ( jsonObj . hasOwnProperty ( "eip155" ) )
2024-07-29 19:46:03 +03:00
const eip155 = jsonObj . eip155
2024-07-04 16:32:04 +03:00
verify ( eip155 . hasOwnProperty ( "chains" ) )
2024-07-29 19:46:03 +03:00
const chains = eip155 . chains
2024-07-04 16:32:04 +03:00
verify ( chains . length === 2 )
verify ( chains [ 0 ] === "eip155:1" )
verify ( chains [ 1 ] === "eip155:2" )
verify ( eip155 . hasOwnProperty ( "accounts" ) )
2024-07-29 19:46:03 +03:00
const accounts = eip155 . accounts
2024-07-04 16:32:04 +03:00
verify ( accounts . length === 4 )
for ( let chainI = 0 ; chainI < chainsModel . count ; chainI ++ ) {
for ( let accountI = 0 ; accountI < chainsModel . count ; accountI ++ ) {
var found = false
2024-07-29 19:46:03 +03:00
for ( const entry of accounts ) {
2024-07-04 16:32:04 +03:00
if ( entry === ` eip155: $ { chainsModel . get ( chainI ) . chainId } : $ { accountsModel . get ( accountI ) . address } ` ) {
found = true
break
}
}
verify ( found , ` found $ { accountsModel . get ( accountI ) . address } for chain $ { chainsModel . get ( chainI ) . chainId } ` )
}
}
verify ( eip155 . hasOwnProperty ( "methods" ) )
verify ( eip155 . methods . length > 0 )
verify ( eip155 . hasOwnProperty ( "events" ) )
2024-07-22 14:33:47 +03:00
compare ( eip155 . events . length , 2 )
2024-07-04 16:32:04 +03:00
}
2024-07-23 17:26:55 +03:00
function test_filterActiveSessionsForKnownAccounts ( ) {
const account1 = accountsModel . get ( 0 )
const account2 = accountsModel . get ( 1 )
const chainIds = [ chainsModel . get ( 0 ) . chainId , chainsModel . get ( 1 ) . chainId ]
const knownSession = JSON . parse ( Testing . formatApproveSessionResponse ( chainIds , [ account2 . address ] ) )
// Allow the unlikely unknown accounts to cover for the deleted accounts case
const unknownSessionWithKnownAccount = JSON . parse ( Testing . formatApproveSessionResponse ( chainIds , [ '0x03acc' , account1 . address ] ) )
const unknownSession1 = JSON . parse ( Testing . formatApproveSessionResponse ( chainIds , [ '0x83acc' ] ) )
const unknownSession2 = JSON . parse ( Testing . formatApproveSessionResponse ( chainIds , [ '0x12acc' ] ) )
let testSessions = {
"b536a" : knownSession ,
"b537b" : unknownSession1 ,
"b538c" : unknownSession2 ,
"b539d" : unknownSessionWithKnownAccount
}
2024-07-29 19:46:03 +03:00
const res = DAppsHelpers . filterActiveSessionsForKnownAccounts ( testSessions , accountsModel )
2024-07-23 17:26:55 +03:00
compare ( Object . keys ( res ) . length , 2 , "expected two sessions to be returned" )
// Also test that order is stable
compare ( res [ "b536a" ] , knownSession , "expected the known session to be returned" )
compare ( res [ "b539d" ] , unknownSessionWithKnownAccount , "expected the known session to be returned" )
}
2024-07-04 16:32:04 +03:00
}
Component {
id: componentUnderTest
DAppsWorkflow {
2024-07-12 00:00:15 +03:00
loginType: Constants . LoginType . Password
2024-07-04 16:32:04 +03:00
}
}
// TODO #15151: this TestCase if placed before ServiceHelpers was not run with `when: windowShown`. Check if related to the CI crash
TestCase {
id: dappsWorkflowTest
name: "DAppsWorkflow"
when: windowShown
property DAppsWorkflow controlUnderTest: null
SignalSpy {
id: dappsListReadySpy
target: dappsWorkflowTest . controlUnderTest
signalName: "dappsListReady"
}
SignalSpy {
id: pairWCReadySpy
target: dappsWorkflowTest . controlUnderTest
signalName: "pairWCReady"
}
function init ( ) {
let walletStore = createTemporaryObject ( walletStoreComponent , root )
verify ( ! ! walletStore )
let sdk = createTemporaryObject ( sdkComponent , root , { projectId: "12ab" } )
verify ( ! ! sdk )
let store = createTemporaryObject ( dappsStoreComponent , root )
verify ( ! ! store )
2024-06-29 23:24:05 +03:00
let service = createTemporaryObject ( serviceComponent , root , { wcSDK: sdk , store: store , walletRootStore: walletStore } )
2024-07-04 16:32:04 +03:00
verify ( ! ! service )
controlUnderTest = createTemporaryObject ( componentUnderTest , root , { wcService: service } )
verify ( ! ! controlUnderTest )
}
function cleanup ( ) {
dappsListReadySpy . clear ( )
pairWCReadySpy . clear ( )
}
function test_OpenAndCloseDappList ( ) {
waitForRendering ( controlUnderTest )
compare ( dappsListReadySpy . count , 0 , "expected NO dappsListReady signal to be emitted" )
mouseClick ( controlUnderTest )
waitForRendering ( controlUnderTest )
compare ( dappsListReadySpy . count , 1 , "expected dappsListReady signal to be emitted" )
let popup = findChild ( controlUnderTest , "dappsListPopup" )
verify ( ! ! popup )
verify ( popup . opened )
popup . close ( )
waitForRendering ( controlUnderTest )
verify ( ! popup . opened )
}
function test_OpenPairModal ( ) {
waitForRendering ( controlUnderTest )
mouseClick ( controlUnderTest )
waitForRendering ( controlUnderTest )
let popup = findChild ( controlUnderTest , "dappsListPopup" )
verify ( ! ! popup )
verify ( popup . opened )
let connectButton = findChild ( popup , "connectDappButton" )
verify ( ! ! connectButton )
verify ( pairWCReadySpy . count === 0 , "expected NO pairWCReady signal to be emitted" )
mouseClick ( connectButton )
waitForRendering ( controlUnderTest )
verify ( pairWCReadySpy . count === 1 , "expected pairWCReady signal to be emitted" )
let pairWCModal = findChild ( controlUnderTest , "pairWCModal" )
verify ( ! ! pairWCModal )
}
Component {
id: sessionRequestComponent
SessionRequestResolved {
}
}
function test_OpenDappRequestModal ( ) {
waitForRendering ( controlUnderTest )
let service = controlUnderTest . wcService
2024-06-29 23:24:05 +03:00
let td = mockSessionRequestEvent ( this , service . wcSDK , service . walletRootStore . nonWatchAccounts , service . walletRootStore . filteredFlatModel )
2024-07-04 16:32:04 +03:00
waitForRendering ( controlUnderTest )
let popup = findChild ( controlUnderTest , "dappsRequestModal" )
verify ( ! ! popup )
verify ( popup . opened )
verify ( popup . visible )
compare ( popup . dappName , td . session . peer . metadata . name )
2024-07-12 00:00:15 +03:00
compare ( popup . accountName , td . account . name )
compare ( popup . accountAddress , td . account . address )
compare ( popup . networkName , td . network . chainName )
2024-07-04 16:32:04 +03:00
popup . close ( )
waitForRendering ( controlUnderTest )
verify ( ! popup . opened )
verify ( ! popup . visible )
}
2024-07-15 20:30:27 +03:00
function showRequestModal ( ) {
2024-07-04 16:32:04 +03:00
waitForRendering ( controlUnderTest )
let service = controlUnderTest . wcService
2024-06-29 23:24:05 +03:00
let td = mockSessionRequestEvent ( this , service . wcSDK , service . walletRootStore . nonWatchAccounts , service . walletRootStore . filteredFlatModel )
2024-07-04 16:32:04 +03:00
waitForRendering ( controlUnderTest )
2024-07-15 20:30:27 +03:00
td . popup = findChild ( controlUnderTest , "dappsRequestModal" )
verify ( td . popup . opened )
return td
}
function test_RejectDappRequestModal ( ) {
let td = showRequestModal ( )
2024-07-04 16:32:04 +03:00
2024-07-15 20:30:27 +03:00
let rejectButton = findChild ( td . popup , "rejectButton" )
2024-07-04 16:32:04 +03:00
mouseClick ( rejectButton )
compare ( td . sdk . rejectSessionRequestCalls . length , 1 , "expected a call to service.rejectSessionRequest" )
2024-07-15 20:30:27 +03:00
compare ( td . sdk . acceptSessionRequestCalls . length , 0 , "expected no call to service.acceptSessionRequest" )
let store = controlUnderTest . wcService . store
compare ( store . authenticateUserCalls . length , 0 , "expected no call to store.authenticateUser for rejection" )
2024-07-04 16:32:04 +03:00
let args = td . sdk . rejectSessionRequestCalls [ 0 ]
compare ( args . topic , td . topic , "expected topic to be set" )
compare ( args . id , td . request . id , "expected id to be set" )
compare ( args . error , false , "expected no error; it was user rejected" )
waitForRendering ( controlUnderTest )
2024-07-15 20:30:27 +03:00
verify ( ! td . popup . opened )
verify ( ! td . popup . visible )
}
function test_AcceptDappRequestModal ( ) {
let td = showRequestModal ( )
let signButton = findChild ( td . popup , "signButton" )
mouseClick ( signButton )
let store = controlUnderTest . wcService . store
compare ( store . authenticateUserCalls . length , 1 , "expected a call to store.authenticateUser" )
compare ( td . sdk . rejectSessionRequestCalls . length , 0 , "regression, expected no call to service.rejectSessionRequest" )
waitForRendering ( controlUnderTest )
verify ( ! td . popup . opened )
verify ( ! td . popup . visible )
2024-07-04 16:32:04 +03:00
}
}
2024-07-17 19:36:32 +03:00
function mockSessionRequestEvent ( tc , sdk , accountsModel , networksModel ) {
2024-07-29 19:46:03 +03:00
const account = accountsModel . get ( 1 )
const network = networksModel . get ( 1 )
const method = "personal_sign"
const message = "hello world"
const params = [ ` "${DAppsHelpers.strToHex(message)}" ` , ` "${account.address}" ` ]
const topic = "b536a"
const requestEvent = JSON . parse ( Testing . formatSessionRequest ( network . chainId , method , params , topic ) )
const request = tc . createTemporaryObject ( sessionRequestComponent , root , {
event: requestEvent ,
topic ,
id: requestEvent.id ,
method: Constants . personal_sign ,
account ,
network ,
data: message ,
preparedData: message
2024-07-04 16:32:04 +03:00
} )
// Expect to have calls to getActiveSessions from service initialization
2024-07-29 19:46:03 +03:00
const prevRequests = sdk . getActiveSessionsCallbacks . length
2024-07-04 16:32:04 +03:00
sdk . sessionRequestEvent ( requestEvent )
// Service might trigger a sessionRequest event following the getActiveSessions call
2024-07-29 19:46:03 +03:00
const callback = sdk . getActiveSessionsCallbacks [ prevRequests ] . callback
const session = JSON . parse ( Testing . formatApproveSessionResponse ( [ network . chainId , 7 ] , [ account . address ] ) )
2024-07-04 16:32:04 +03:00
callback ( { "b536a" : session } )
return { sdk , session , account , network , topic , request }
}
2024-04-30 14:08:37 +02:00
}