chore: add wallet client (#2462)
* chore: migrate to wallet client * fix: bump client with new esm release version * fix: add new no client error * fix: bump wallet client to 0.1.2 and remove empty params from method calls * fix: bump wallet client * fix: format * fix: reset client on url change * fix: trading-deal-ticket tests after wallet client added * fix: amend and cancel order tests, global connect tests * chore: fix typescript error * fix: connect wallet before mobile view test for console-lite Co-authored-by: Matthew Russell <mattrussell36@gmail.com>
This commit is contained in:
parent
0d1c9ff0f1
commit
aac3799b14
@ -67,6 +67,7 @@ describe('market selector', { tags: '@smoke' }, () => {
|
|||||||
it('mobile view', () => {
|
it('mobile view', () => {
|
||||||
cy.viewport('iphone-xr');
|
cy.viewport('iphone-xr');
|
||||||
cy.visit(`/trading/${marketId}`);
|
cy.visit(`/trading/${marketId}`);
|
||||||
|
cy.connectVegaWallet();
|
||||||
cy.get('[role="dialog"]').should('not.exist');
|
cy.get('[role="dialog"]').should('not.exist');
|
||||||
cy.getByTestId('arrow-button').click();
|
cy.getByTestId('arrow-button').click();
|
||||||
cy.get('[role="dialog"]').should('be.visible');
|
cy.get('[role="dialog"]').should('be.visible');
|
||||||
|
@ -30,11 +30,20 @@ const displayTomorrow = () => {
|
|||||||
|
|
||||||
describe('time in force default values', () => {
|
describe('time in force default values', () => {
|
||||||
before(() => {
|
before(() => {
|
||||||
|
cy.window().then((win) => {
|
||||||
|
win.localStorage.setItem(
|
||||||
|
'vega_wallet_config',
|
||||||
|
JSON.stringify({
|
||||||
|
token: Cypress.env('VEGA_WALLET_API_TOKEN'),
|
||||||
|
connector: 'jsonRpc',
|
||||||
|
url: 'http://localhost:1789',
|
||||||
|
})
|
||||||
|
);
|
||||||
|
});
|
||||||
cy.mockTradingPage();
|
cy.mockTradingPage();
|
||||||
cy.mockSubscription();
|
cy.mockSubscription();
|
||||||
cy.visit('/#/markets/market-0');
|
cy.visit('/#/markets/market-0');
|
||||||
cy.wait('@Market');
|
cy.wait('@Market');
|
||||||
cy.connectVegaWallet();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('must have market order set up to IOC by default', function () {
|
it('must have market order set up to IOC by default', function () {
|
||||||
@ -59,20 +68,15 @@ describe('time in force default values', () => {
|
|||||||
describe('must submit order', { tags: '@smoke' }, () => {
|
describe('must submit order', { tags: '@smoke' }, () => {
|
||||||
// 7002-SORD-039
|
// 7002-SORD-039
|
||||||
before(() => {
|
before(() => {
|
||||||
|
setVegaConfig();
|
||||||
cy.mockTradingPage();
|
cy.mockTradingPage();
|
||||||
cy.mockSubscription();
|
cy.mockSubscription();
|
||||||
cy.visit('/#/markets/market-0');
|
cy.visit('/#/markets/market-0');
|
||||||
cy.wait('@Market');
|
cy.wait('@Market');
|
||||||
cy.connectVegaWallet();
|
|
||||||
cy.window().then(function (window) {
|
|
||||||
cy.wrap(window.localStorage.getItem('vega_wallet_config')).as('cfg');
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
cy.window().then(function (window) {
|
setVegaConfig();
|
||||||
window.localStorage.setItem('vega_wallet_config', this.cfg);
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('successfully places market buy order', () => {
|
it('successfully places market buy order', () => {
|
||||||
@ -158,6 +162,7 @@ describe(
|
|||||||
{ tags: '@regression' },
|
{ tags: '@regression' },
|
||||||
() => {
|
() => {
|
||||||
before(() => {
|
before(() => {
|
||||||
|
setVegaConfig();
|
||||||
cy.mockTradingPage(
|
cy.mockTradingPage(
|
||||||
Schema.MarketState.STATE_SUSPENDED,
|
Schema.MarketState.STATE_SUSPENDED,
|
||||||
Schema.MarketTradingMode.TRADING_MODE_BATCH_AUCTION,
|
Schema.MarketTradingMode.TRADING_MODE_BATCH_AUCTION,
|
||||||
@ -166,16 +171,10 @@ describe(
|
|||||||
cy.mockSubscription();
|
cy.mockSubscription();
|
||||||
cy.visit('/#/markets/market-0');
|
cy.visit('/#/markets/market-0');
|
||||||
cy.wait('@Market');
|
cy.wait('@Market');
|
||||||
cy.connectVegaWallet();
|
|
||||||
cy.window().then(function (window) {
|
|
||||||
cy.wrap(window.localStorage.getItem('vega_wallet_config')).as('cfg');
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
cy.window().then(function (window) {
|
setVegaConfig();
|
||||||
window.localStorage.setItem('vega_wallet_config', this.cfg);
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('successfully places limit buy order', () => {
|
it('successfully places limit buy order', () => {
|
||||||
@ -232,6 +231,7 @@ describe(
|
|||||||
{ tags: '@regression' },
|
{ tags: '@regression' },
|
||||||
() => {
|
() => {
|
||||||
before(() => {
|
before(() => {
|
||||||
|
setVegaConfig();
|
||||||
cy.mockTradingPage(
|
cy.mockTradingPage(
|
||||||
Schema.MarketState.STATE_SUSPENDED,
|
Schema.MarketState.STATE_SUSPENDED,
|
||||||
Schema.MarketTradingMode.TRADING_MODE_OPENING_AUCTION,
|
Schema.MarketTradingMode.TRADING_MODE_OPENING_AUCTION,
|
||||||
@ -240,16 +240,10 @@ describe(
|
|||||||
cy.mockSubscription();
|
cy.mockSubscription();
|
||||||
cy.visit('/#/markets/market-0');
|
cy.visit('/#/markets/market-0');
|
||||||
cy.wait('@Market');
|
cy.wait('@Market');
|
||||||
cy.connectVegaWallet();
|
|
||||||
cy.window().then(function (window) {
|
|
||||||
cy.wrap(window.localStorage.getItem('vega_wallet_config')).as('cfg');
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
cy.window().then(function (window) {
|
setVegaConfig();
|
||||||
window.localStorage.setItem('vega_wallet_config', this.cfg);
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('successfully places limit buy order', () => {
|
it('successfully places limit buy order', () => {
|
||||||
@ -306,6 +300,7 @@ describe(
|
|||||||
{ tags: '@regression' },
|
{ tags: '@regression' },
|
||||||
() => {
|
() => {
|
||||||
before(() => {
|
before(() => {
|
||||||
|
setVegaConfig();
|
||||||
cy.mockTradingPage(
|
cy.mockTradingPage(
|
||||||
Schema.MarketState.STATE_SUSPENDED,
|
Schema.MarketState.STATE_SUSPENDED,
|
||||||
Schema.MarketTradingMode.TRADING_MODE_MONITORING_AUCTION,
|
Schema.MarketTradingMode.TRADING_MODE_MONITORING_AUCTION,
|
||||||
@ -314,16 +309,10 @@ describe(
|
|||||||
cy.mockSubscription();
|
cy.mockSubscription();
|
||||||
cy.visit('/#/markets/market-0');
|
cy.visit('/#/markets/market-0');
|
||||||
cy.wait('@Market');
|
cy.wait('@Market');
|
||||||
cy.connectVegaWallet();
|
|
||||||
cy.window().then(function (window) {
|
|
||||||
cy.wrap(window.localStorage.getItem('vega_wallet_config')).as('cfg');
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
cy.window().then(function (window) {
|
setVegaConfig();
|
||||||
window.localStorage.setItem('vega_wallet_config', this.cfg);
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('successfully places limit buy order', () => {
|
it('successfully places limit buy order', () => {
|
||||||
@ -421,10 +410,10 @@ describe('deal ticket validation', { tags: '@smoke' }, () => {
|
|||||||
|
|
||||||
describe('deal ticket size validation', { tags: '@smoke' }, function () {
|
describe('deal ticket size validation', { tags: '@smoke' }, function () {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
|
setVegaConfig();
|
||||||
cy.mockTradingPage();
|
cy.mockTradingPage();
|
||||||
cy.visit('/#/markets/market-0');
|
cy.visit('/#/markets/market-0');
|
||||||
cy.wait('@Market');
|
cy.wait('@Market');
|
||||||
cy.connectVegaWallet();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('must warn if order size input has too many digits after the decimal place', function () {
|
it('must warn if order size input has too many digits after the decimal place', function () {
|
||||||
@ -455,14 +444,18 @@ describe('deal ticket size validation', { tags: '@smoke' }, function () {
|
|||||||
|
|
||||||
describe('limit order validations', { tags: '@smoke' }, () => {
|
describe('limit order validations', { tags: '@smoke' }, () => {
|
||||||
before(() => {
|
before(() => {
|
||||||
|
setVegaConfig();
|
||||||
cy.mockTradingPage();
|
cy.mockTradingPage();
|
||||||
cy.mockSubscription();
|
cy.mockSubscription();
|
||||||
cy.visit('/#/markets/market-0');
|
cy.visit('/#/markets/market-0');
|
||||||
cy.connectVegaWallet();
|
|
||||||
cy.wait('@Market');
|
cy.wait('@Market');
|
||||||
cy.getByTestId(toggleLimit).click();
|
cy.getByTestId(toggleLimit).click();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
setVegaConfig();
|
||||||
|
});
|
||||||
|
|
||||||
it('must see the price unit', function () {
|
it('must see the price unit', function () {
|
||||||
//7002-SORD-018
|
//7002-SORD-018
|
||||||
cy.getByTestId(orderPriceField)
|
cy.getByTestId(orderPriceField)
|
||||||
@ -544,11 +537,17 @@ describe('limit order validations', { tags: '@smoke' }, () => {
|
|||||||
|
|
||||||
describe('market order validations', { tags: '@smoke' }, () => {
|
describe('market order validations', { tags: '@smoke' }, () => {
|
||||||
before(() => {
|
before(() => {
|
||||||
|
setVegaConfig();
|
||||||
cy.mockTradingPage();
|
cy.mockTradingPage();
|
||||||
cy.visit('/#/markets/market-0');
|
cy.visit('/#/markets/market-0');
|
||||||
cy.wait('@Market');
|
cy.wait('@Market');
|
||||||
cy.getByTestId(toggleMarket).click();
|
cy.getByTestId(toggleMarket).click();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
setVegaConfig();
|
||||||
|
});
|
||||||
|
|
||||||
it('must not see the price unit', function () {
|
it('must not see the price unit', function () {
|
||||||
//7002-SORD-019
|
//7002-SORD-019
|
||||||
cy.getByTestId(orderPriceField).should('not.exist');
|
cy.getByTestId(orderPriceField).should('not.exist');
|
||||||
@ -587,6 +586,7 @@ describe('market order validations', { tags: '@smoke' }, () => {
|
|||||||
|
|
||||||
describe('suspended market validation', { tags: '@regression' }, () => {
|
describe('suspended market validation', { tags: '@regression' }, () => {
|
||||||
before(() => {
|
before(() => {
|
||||||
|
setVegaConfig();
|
||||||
cy.mockTradingPage(
|
cy.mockTradingPage(
|
||||||
Schema.MarketState.STATE_SUSPENDED,
|
Schema.MarketState.STATE_SUSPENDED,
|
||||||
Schema.MarketTradingMode.TRADING_MODE_MONITORING_AUCTION,
|
Schema.MarketTradingMode.TRADING_MODE_MONITORING_AUCTION,
|
||||||
@ -595,7 +595,10 @@ describe('suspended market validation', { tags: '@regression' }, () => {
|
|||||||
cy.mockSubscription();
|
cy.mockSubscription();
|
||||||
cy.visit('/#/markets/market-0');
|
cy.visit('/#/markets/market-0');
|
||||||
cy.wait('@Market');
|
cy.wait('@Market');
|
||||||
cy.connectVegaWallet();
|
});
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
setVegaConfig();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should show warning for market order', function () {
|
it('should show warning for market order', function () {
|
||||||
@ -608,6 +611,7 @@ describe('suspended market validation', { tags: '@regression' }, () => {
|
|||||||
'This market is in auction until it reaches sufficient liquidity. Only limit orders are permitted when market is in auction'
|
'This market is in auction until it reaches sufficient liquidity. Only limit orders are permitted when market is in auction'
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should show info for allowed TIF', function () {
|
it('should show info for allowed TIF', function () {
|
||||||
cy.getByTestId(toggleLimit).click();
|
cy.getByTestId(toggleLimit).click();
|
||||||
cy.getByTestId(orderPriceField).clear().type('0.1');
|
cy.getByTestId(orderPriceField).clear().type('0.1');
|
||||||
@ -635,6 +639,7 @@ describe('suspended market validation', { tags: '@regression' }, () => {
|
|||||||
describe('account validation', { tags: '@regression' }, () => {
|
describe('account validation', { tags: '@regression' }, () => {
|
||||||
describe('zero balance error', () => {
|
describe('zero balance error', () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
|
setVegaConfig();
|
||||||
cy.mockTradingPage();
|
cy.mockTradingPage();
|
||||||
cy.mockGQL((req) => {
|
cy.mockGQL((req) => {
|
||||||
aliasGQLQuery(
|
aliasGQLQuery(
|
||||||
@ -663,7 +668,6 @@ describe('account validation', { tags: '@regression' }, () => {
|
|||||||
});
|
});
|
||||||
cy.mockSubscription();
|
cy.mockSubscription();
|
||||||
cy.visit('/#/markets/market-0');
|
cy.visit('/#/markets/market-0');
|
||||||
cy.connectVegaWallet();
|
|
||||||
cy.wait('@Market');
|
cy.wait('@Market');
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -682,6 +686,7 @@ describe('account validation', { tags: '@regression' }, () => {
|
|||||||
|
|
||||||
describe('not enough balance warning', () => {
|
describe('not enough balance warning', () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
|
setVegaConfig();
|
||||||
cy.mockTradingPage();
|
cy.mockTradingPage();
|
||||||
cy.mockGQL((req) => {
|
cy.mockGQL((req) => {
|
||||||
aliasGQLQuery(
|
aliasGQLQuery(
|
||||||
@ -699,9 +704,9 @@ describe('account validation', { tags: '@regression' }, () => {
|
|||||||
});
|
});
|
||||||
cy.mockSubscription();
|
cy.mockSubscription();
|
||||||
cy.visit('/#/markets/market-0');
|
cy.visit('/#/markets/market-0');
|
||||||
cy.connectVegaWallet();
|
|
||||||
cy.wait('@Market');
|
cy.wait('@Market');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should display info and button for deposit', () => {
|
it('should display info and button for deposit', () => {
|
||||||
//7002-SORD-003
|
//7002-SORD-003
|
||||||
// warning should show immediately
|
// warning should show immediately
|
||||||
@ -740,3 +745,16 @@ const createOrder = (order: OrderSubmission): void => {
|
|||||||
}
|
}
|
||||||
cy.getByTestId(placeOrderBtn).click();
|
cy.getByTestId(placeOrderBtn).click();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const setVegaConfig = () => {
|
||||||
|
cy.window().then((win) => {
|
||||||
|
win.localStorage.setItem(
|
||||||
|
'vega_wallet_config',
|
||||||
|
JSON.stringify({
|
||||||
|
token: Cypress.env('VEGA_WALLET_API_TOKEN'),
|
||||||
|
connector: 'jsonRpc',
|
||||||
|
url: 'http://localhost:1789',
|
||||||
|
})
|
||||||
|
);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
@ -1,11 +0,0 @@
|
|||||||
import { ethers } from 'ethers';
|
|
||||||
import type { Transaction } from '@vegaprotocol/wallet';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Base64 encode a transaction object
|
|
||||||
*/
|
|
||||||
export const encodeTransaction = (tx: Transaction): string => {
|
|
||||||
return ethers.utils.base64.encode(
|
|
||||||
ethers.utils.toUtf8Bytes(JSON.stringify(tx))
|
|
||||||
);
|
|
||||||
};
|
|
@ -7,7 +7,6 @@ import type {
|
|||||||
OrderSubmissionBody,
|
OrderSubmissionBody,
|
||||||
Transaction,
|
Transaction,
|
||||||
} from '@vegaprotocol/wallet';
|
} from '@vegaprotocol/wallet';
|
||||||
import { encodeTransaction } from './encode-transaction';
|
|
||||||
|
|
||||||
export const testOrderSubmission = (
|
export const testOrderSubmission = (
|
||||||
order: OrderSubmission,
|
order: OrderSubmission,
|
||||||
@ -18,9 +17,6 @@ export const testOrderSubmission = (
|
|||||||
...expected,
|
...expected,
|
||||||
};
|
};
|
||||||
|
|
||||||
expectedOrder.expiresAt = expectedOrder.expiresAt || undefined;
|
|
||||||
expectedOrder.price = expectedOrder.price || undefined;
|
|
||||||
|
|
||||||
const transaction: OrderSubmissionBody = {
|
const transaction: OrderSubmissionBody = {
|
||||||
orderSubmission: expectedOrder,
|
orderSubmission: expectedOrder,
|
||||||
};
|
};
|
||||||
@ -36,9 +32,6 @@ export const testOrderAmendment = (
|
|||||||
...expected,
|
...expected,
|
||||||
};
|
};
|
||||||
|
|
||||||
expectedOrder.expiresAt = expectedOrder.expiresAt || undefined;
|
|
||||||
expectedOrder.price = expectedOrder.price || undefined;
|
|
||||||
|
|
||||||
const transaction: OrderAmendmentBody = {
|
const transaction: OrderAmendmentBody = {
|
||||||
orderAmendment: expectedOrder,
|
orderAmendment: expectedOrder,
|
||||||
};
|
};
|
||||||
@ -70,7 +63,7 @@ const vegaWalletTransaction = (transaction: Transaction) => {
|
|||||||
?.token,
|
?.token,
|
||||||
publicKey: Cypress.env('VEGA_PUBLIC_KEY2'),
|
publicKey: Cypress.env('VEGA_PUBLIC_KEY2'),
|
||||||
sendingMode: 'TYPE_SYNC',
|
sendingMode: 'TYPE_SYNC',
|
||||||
encodedTransaction: encodeTransaction(transaction),
|
transaction,
|
||||||
});
|
});
|
||||||
cy.getByTestId(dialogTitle).should(
|
cy.getByTestId(dialogTitle).should(
|
||||||
'have.text',
|
'have.text',
|
||||||
|
@ -1,74 +1,11 @@
|
|||||||
import { t } from '@vegaprotocol/react-helpers';
|
import { t } from '@vegaprotocol/react-helpers';
|
||||||
import { z } from 'zod';
|
import { WalletClient } from '@vegaprotocol/wallet-client';
|
||||||
import { clearConfig, getConfig, setConfig } from '../storage';
|
import { clearConfig, getConfig, setConfig } from '../storage';
|
||||||
import { encodeTransaction } from '../utils';
|
|
||||||
import type { Transaction, VegaConnector } from './vega-connector';
|
import type { Transaction, VegaConnector } from './vega-connector';
|
||||||
import { WalletError } from './vega-connector';
|
import { WalletError } from './vega-connector';
|
||||||
|
|
||||||
const VERSION = 'v2';
|
const VERSION = 'v2';
|
||||||
|
|
||||||
enum Methods {
|
|
||||||
ConnectWallet = 'client.connect_wallet',
|
|
||||||
DisconnectWallet = 'client.disconnect_wallet',
|
|
||||||
ListKeys = 'client.list_keys',
|
|
||||||
SendTransaction = 'client.send_transaction',
|
|
||||||
GetChainId = 'client.get_chain_id',
|
|
||||||
}
|
|
||||||
|
|
||||||
const BaseSchema = z.object({
|
|
||||||
id: z.string(),
|
|
||||||
jsonrpc: z.literal('2.0'),
|
|
||||||
});
|
|
||||||
|
|
||||||
const ConnectWalletSchema = BaseSchema.extend({
|
|
||||||
result: z.object({
|
|
||||||
token: z.string(),
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
|
|
||||||
const ListKeysSchema = BaseSchema.extend({
|
|
||||||
result: z.object({
|
|
||||||
keys: z.array(
|
|
||||||
z.object({
|
|
||||||
publicKey: z.string(),
|
|
||||||
name: z.string(),
|
|
||||||
})
|
|
||||||
),
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
|
|
||||||
const GetChainIdSchema = BaseSchema.extend({
|
|
||||||
result: z.object({
|
|
||||||
chainID: z.string(),
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
|
|
||||||
const SendTransactionSchema = BaseSchema.extend({
|
|
||||||
result: z.object({
|
|
||||||
receivedAt: z.string(),
|
|
||||||
sentAt: z.string(),
|
|
||||||
transactionHash: z.string(),
|
|
||||||
transaction: z.object({
|
|
||||||
signature: z.object({
|
|
||||||
value: z.string(),
|
|
||||||
}),
|
|
||||||
}),
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
|
|
||||||
type JsonRpcError = {
|
|
||||||
message: string;
|
|
||||||
code: number;
|
|
||||||
data?: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
type Response =
|
|
||||||
| z.infer<typeof ConnectWalletSchema>
|
|
||||||
| z.infer<typeof ListKeysSchema>
|
|
||||||
| z.infer<typeof GetChainIdSchema>
|
|
||||||
| z.infer<typeof SendTransactionSchema>
|
|
||||||
| { error: JsonRpcError };
|
|
||||||
|
|
||||||
export const ClientErrors = {
|
export const ClientErrors = {
|
||||||
NO_SERVICE: new WalletError(t('No service'), 100),
|
NO_SERVICE: new WalletError(t('No service'), 100),
|
||||||
NO_TOKEN: new WalletError(t('No token'), 101),
|
NO_TOKEN: new WalletError(t('No token'), 101),
|
||||||
@ -86,51 +23,69 @@ export const ClientErrors = {
|
|||||||
),
|
),
|
||||||
} as const;
|
} as const;
|
||||||
|
|
||||||
|
class NoClientError extends Error {
|
||||||
|
constructor() {
|
||||||
|
super(
|
||||||
|
t('No client found. The connector needs to be initialized with a url.')
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export class JsonRpcConnector implements VegaConnector {
|
export class JsonRpcConnector implements VegaConnector {
|
||||||
version = VERSION;
|
version = VERSION;
|
||||||
url: string | null = null;
|
private _url: string | null = null;
|
||||||
token: string | null = null;
|
token: string | null = null;
|
||||||
reqId = 0;
|
reqId = 0;
|
||||||
|
client?: WalletClient;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
const cfg = getConfig();
|
const cfg = getConfig();
|
||||||
if (cfg) {
|
|
||||||
this.token = cfg.token;
|
this.token = cfg?.token ?? null;
|
||||||
|
|
||||||
|
if (cfg && cfg.url) {
|
||||||
this.url = cfg.url;
|
this.url = cfg.url;
|
||||||
|
this.client = new WalletClient({
|
||||||
|
address: cfg.url,
|
||||||
|
token: cfg.token ?? undefined,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
set url(url: string) {
|
||||||
|
this._url = url;
|
||||||
|
this.client = new WalletClient({
|
||||||
|
address: url,
|
||||||
|
token: this.token ?? undefined,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
async getChainId() {
|
async getChainId() {
|
||||||
const result = await this.request(Methods.GetChainId);
|
if (!this.client) {
|
||||||
if ('error' in result) {
|
throw new NoClientError();
|
||||||
throw this.wrapError(result.error);
|
|
||||||
}
|
}
|
||||||
const parseResult = GetChainIdSchema.safeParse(result);
|
try {
|
||||||
if (parseResult.success) {
|
const { result } = await this.client.GetChainId();
|
||||||
return parseResult.data.result;
|
return result;
|
||||||
} else {
|
} catch (err) {
|
||||||
throw ClientErrors.INVALID_RESPONSE;
|
throw ClientErrors.INVALID_RESPONSE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async connectWallet() {
|
async connectWallet() {
|
||||||
const result = await this.request(Methods.ConnectWallet, {
|
if (!this.client) {
|
||||||
hostname: window.location.host,
|
throw new NoClientError();
|
||||||
});
|
|
||||||
if ('error' in result) {
|
|
||||||
throw this.wrapError(result.error);
|
|
||||||
}
|
}
|
||||||
const parseResult = ConnectWalletSchema.safeParse(result);
|
|
||||||
|
|
||||||
if (parseResult.success) {
|
try {
|
||||||
// store token and other config for eager connect and subsequent requests
|
const { result } = await this.client.ConnectWallet();
|
||||||
setConfig({
|
setConfig({
|
||||||
token: parseResult.data.result.token,
|
token: result.token,
|
||||||
connector: 'jsonRpc',
|
connector: 'jsonRpc',
|
||||||
url: this.url,
|
url: this.url,
|
||||||
});
|
});
|
||||||
return parseResult.data.result;
|
return result;
|
||||||
} else {
|
} catch (err) {
|
||||||
throw ClientErrors.INVALID_RESPONSE;
|
throw ClientErrors.INVALID_RESPONSE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -138,80 +93,57 @@ export class JsonRpcConnector implements VegaConnector {
|
|||||||
// connect actually calling list_keys here, not to be confused with connect_wallet
|
// connect actually calling list_keys here, not to be confused with connect_wallet
|
||||||
// which retrieves the session token
|
// which retrieves the session token
|
||||||
async connect() {
|
async connect() {
|
||||||
const cfg = getConfig();
|
if (!this.client) {
|
||||||
if (!cfg?.token) {
|
throw new NoClientError();
|
||||||
throw ClientErrors.NO_TOKEN;
|
|
||||||
}
|
}
|
||||||
const result = await this.request(Methods.ListKeys, {
|
|
||||||
token: cfg.token,
|
try {
|
||||||
});
|
const { result } = await this.client.ListKeys();
|
||||||
if ('error' in result) {
|
return result.keys;
|
||||||
throw this.wrapError(result.error);
|
} catch (err) {
|
||||||
}
|
|
||||||
const parseResult = ListKeysSchema.safeParse(result);
|
|
||||||
if (parseResult.success) {
|
|
||||||
return parseResult.data.result.keys;
|
|
||||||
} else {
|
|
||||||
throw ClientErrors.INVALID_RESPONSE;
|
throw ClientErrors.INVALID_RESPONSE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async disconnect() {
|
async disconnect() {
|
||||||
const cfg = getConfig();
|
if (!this.client) {
|
||||||
|
throw new NoClientError();
|
||||||
if (cfg?.token) {
|
|
||||||
await this.request(Methods.DisconnectWallet, {
|
|
||||||
token: cfg.token,
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await this.client.DisconnectWallet();
|
||||||
clearConfig();
|
clearConfig();
|
||||||
}
|
}
|
||||||
|
|
||||||
async sendTx(pubKey: string, transaction: Transaction) {
|
async sendTx(pubKey: string, transaction: Transaction) {
|
||||||
const cfg = getConfig();
|
if (!this.client) {
|
||||||
if (!cfg?.token) {
|
throw new NoClientError();
|
||||||
throw ClientErrors.NO_TOKEN;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const result = await this.request(Methods.SendTransaction, {
|
try {
|
||||||
token: cfg.token,
|
const { result } = await this.client.SendTransaction({
|
||||||
publicKey: pubKey,
|
publicKey: pubKey,
|
||||||
sendingMode: 'TYPE_SYNC',
|
sendingMode: 'TYPE_SYNC',
|
||||||
encodedTransaction: encodeTransaction(transaction),
|
transaction,
|
||||||
});
|
});
|
||||||
|
|
||||||
if ('error' in result) {
|
|
||||||
// In the case of sending a tx, error code 3001 indicates that the
|
|
||||||
// user rejected the tx. Returning null will allow the dialog to close immediately
|
|
||||||
if (result.error.code === 3001) {
|
|
||||||
return null;
|
|
||||||
} else {
|
|
||||||
throw this.wrapError(result.error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const parsedResult = SendTransactionSchema.safeParse(result);
|
|
||||||
|
|
||||||
if (parsedResult.success) {
|
|
||||||
return {
|
return {
|
||||||
transactionHash: parsedResult.data.result.transactionHash,
|
transactionHash: result.transactionHash,
|
||||||
sentAt: parsedResult.data.result.sentAt,
|
sentAt: result.sentAt,
|
||||||
receivedAt: parsedResult.data.result.receivedAt,
|
receivedAt: result.receivedAt,
|
||||||
signature: parsedResult.data.result.transaction.signature.value,
|
signature: result.transaction.signature.value,
|
||||||
};
|
};
|
||||||
} else {
|
} catch (err) {
|
||||||
throw ClientErrors.INVALID_RESPONSE;
|
throw ClientErrors.INVALID_RESPONSE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async checkCompat() {
|
async checkCompat() {
|
||||||
try {
|
try {
|
||||||
const result = await fetch(`${this.url}/api/${this.version}/methods`);
|
const result = await fetch(`${this._url}/api/${this.version}/methods`);
|
||||||
if (!result.ok) {
|
if (!result.ok) {
|
||||||
const err = ClientErrors.INVALID_WALLET;
|
const err = ClientErrors.INVALID_WALLET;
|
||||||
err.data = t(
|
err.data = t(
|
||||||
`The wallet running at ${this.url} is not supported. Required version is ${this.version}`
|
`The wallet running at ${this._url} is not supported. Required version is ${this.version}`
|
||||||
);
|
);
|
||||||
throw err;
|
throw err;
|
||||||
}
|
}
|
||||||
@ -224,29 +156,4 @@ export class JsonRpcConnector implements VegaConnector {
|
|||||||
throw ClientErrors.NO_SERVICE;
|
throw ClientErrors.NO_SERVICE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private async request(method: Methods, params?: object): Promise<Response> {
|
|
||||||
try {
|
|
||||||
const result = await fetch(`${this.url}/api/${this.version}/requests`, {
|
|
||||||
method: 'post',
|
|
||||||
body: JSON.stringify({
|
|
||||||
jsonrpc: '2.0',
|
|
||||||
method,
|
|
||||||
params,
|
|
||||||
id: `${this.reqId++}`,
|
|
||||||
}),
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
const json = await result.json();
|
|
||||||
return json;
|
|
||||||
} catch (err) {
|
|
||||||
throw ClientErrors.NO_SERVICE;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private wrapError(error: JsonRpcError) {
|
|
||||||
return new WalletError(error.message, error.code, error.data);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -17,7 +17,7 @@ export interface VegaWalletContextShape {
|
|||||||
connect: (connector: VegaConnector) => Promise<PubKey[] | null>;
|
connect: (connector: VegaConnector) => Promise<PubKey[] | null>;
|
||||||
|
|
||||||
/** Disconnects from the connector and clears public key state */
|
/** Disconnects from the connector and clears public key state */
|
||||||
disconnect: () => Promise<boolean>;
|
disconnect: () => Promise<void>;
|
||||||
|
|
||||||
/** Sets the current selected public key */
|
/** Sets the current selected public key */
|
||||||
selectPubKey: (pubKey: string) => void;
|
selectPubKey: (pubKey: string) => void;
|
||||||
|
@ -59,23 +59,19 @@ export const VegaWalletProvider = ({ children }: VegaWalletProviderProps) => {
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const disconnect = useCallback(async () => {
|
const disconnect = useCallback(async () => {
|
||||||
|
// always clear state after attempted disconnection.. this
|
||||||
|
// is because long lived token sessions (used in tests)
|
||||||
|
// cannot be cleared. Clearing state will force user to reconnect
|
||||||
|
// again as expected
|
||||||
|
setPubKeys(null);
|
||||||
|
setPubKey(null);
|
||||||
|
LocalStorage.removeItem(WALLET_KEY);
|
||||||
try {
|
try {
|
||||||
await connector.current?.disconnect();
|
await connector.current?.disconnect();
|
||||||
setPubKeys(null);
|
|
||||||
setPubKey(null);
|
|
||||||
connector.current = null;
|
connector.current = null;
|
||||||
LocalStorage.removeItem(WALLET_KEY);
|
|
||||||
return true;
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error(err);
|
console.error(err);
|
||||||
if (err instanceof WalletError && err.code === 100) {
|
connector.current = null;
|
||||||
setPubKeys(null);
|
|
||||||
setPubKey(null);
|
|
||||||
connector.current = null;
|
|
||||||
LocalStorage.removeItem(WALLET_KEY);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
@ -36,6 +36,7 @@
|
|||||||
"@sentry/nextjs": "^6.19.3",
|
"@sentry/nextjs": "^6.19.3",
|
||||||
"@sentry/react": "^6.19.2",
|
"@sentry/react": "^6.19.2",
|
||||||
"@sentry/tracing": "^6.19.2",
|
"@sentry/tracing": "^6.19.2",
|
||||||
|
"@vegaprotocol/wallet-client": "0.1.4",
|
||||||
"@walletconnect/ethereum-provider": "^1.7.5",
|
"@walletconnect/ethereum-provider": "^1.7.5",
|
||||||
"@web3-react/core": "8.0.20-beta.0",
|
"@web3-react/core": "8.0.20-beta.0",
|
||||||
"@web3-react/metamask": "8.0.16-beta.0",
|
"@web3-react/metamask": "8.0.16-beta.0",
|
||||||
|
13
yarn.lock
13
yarn.lock
@ -7559,6 +7559,15 @@
|
|||||||
"@typescript-eslint/types" "5.40.0"
|
"@typescript-eslint/types" "5.40.0"
|
||||||
eslint-visitor-keys "^3.3.0"
|
eslint-visitor-keys "^3.3.0"
|
||||||
|
|
||||||
|
"@vegaprotocol/wallet-client@0.1.4":
|
||||||
|
version "0.1.4"
|
||||||
|
resolved "https://registry.yarnpkg.com/@vegaprotocol/wallet-client/-/wallet-client-0.1.4.tgz#202fa1a84dbef57199810383f2887a7ee0afd64c"
|
||||||
|
integrity sha512-uGEbusoi3lwyl7Nn9ovBg9YHrfcH/Rl33KUcLfdMeTX8FZO+n7BVm3ejd00e5RsM/PJlqJ1oW4qkiq7kervxng==
|
||||||
|
dependencies:
|
||||||
|
express "4.18.2"
|
||||||
|
nanoid "3.3.4"
|
||||||
|
node-fetch "2.6.7"
|
||||||
|
|
||||||
"@walletconnect/browser-utils@^1.8.0":
|
"@walletconnect/browser-utils@^1.8.0":
|
||||||
version "1.8.0"
|
version "1.8.0"
|
||||||
resolved "https://registry.yarnpkg.com/@walletconnect/browser-utils/-/browser-utils-1.8.0.tgz#33c10e777aa6be86c713095b5206d63d32df0951"
|
resolved "https://registry.yarnpkg.com/@walletconnect/browser-utils/-/browser-utils-1.8.0.tgz#33c10e777aa6be86c713095b5206d63d32df0951"
|
||||||
@ -12798,7 +12807,7 @@ expect@^29.0.0:
|
|||||||
jest-message-util "^29.1.2"
|
jest-message-util "^29.1.2"
|
||||||
jest-util "^29.1.2"
|
jest-util "^29.1.2"
|
||||||
|
|
||||||
express@^4.17.1, express@^4.17.3:
|
express@4.18.2, express@^4.17.1, express@^4.17.3:
|
||||||
version "4.18.2"
|
version "4.18.2"
|
||||||
resolved "https://registry.yarnpkg.com/express/-/express-4.18.2.tgz#3fabe08296e930c796c19e3c516979386ba9fd59"
|
resolved "https://registry.yarnpkg.com/express/-/express-4.18.2.tgz#3fabe08296e930c796c19e3c516979386ba9fd59"
|
||||||
integrity sha512-5/PsL6iGPdfQ/lKM1UuielYgv3BUoJfz1aUwU9vHZ+J7gyvwdQXFEBIEIaxeGf0GIcreATNyBExtalisDbuMqQ==
|
integrity sha512-5/PsL6iGPdfQ/lKM1UuielYgv3BUoJfz1aUwU9vHZ+J7gyvwdQXFEBIEIaxeGf0GIcreATNyBExtalisDbuMqQ==
|
||||||
@ -17333,7 +17342,7 @@ nan@^2.12.1:
|
|||||||
resolved "https://registry.yarnpkg.com/nan/-/nan-2.17.0.tgz#c0150a2368a182f033e9aa5195ec76ea41a199cb"
|
resolved "https://registry.yarnpkg.com/nan/-/nan-2.17.0.tgz#c0150a2368a182f033e9aa5195ec76ea41a199cb"
|
||||||
integrity sha512-2ZTgtl0nJsO0KQCjEpxcIr5D+Yv90plTitZt9JBfQvVJDS5seMl3FOvsh3+9CoYWXf/1l5OaZzzF6nDm4cagaQ==
|
integrity sha512-2ZTgtl0nJsO0KQCjEpxcIr5D+Yv90plTitZt9JBfQvVJDS5seMl3FOvsh3+9CoYWXf/1l5OaZzzF6nDm4cagaQ==
|
||||||
|
|
||||||
nanoid@^3.3.1, nanoid@^3.3.4:
|
nanoid@3.3.4, nanoid@^3.3.1, nanoid@^3.3.4:
|
||||||
version "3.3.4"
|
version "3.3.4"
|
||||||
resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.4.tgz#730b67e3cd09e2deacf03c027c81c9d9dbc5e8ab"
|
resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.4.tgz#730b67e3cd09e2deacf03c027c81c9d9dbc5e8ab"
|
||||||
integrity sha512-MqBkQh/OHTS2egovRtLk45wEyNXwF+cokD+1YPf9u5VfJiRdAiRwB2froX5Co9Rh20xs4siNPm8naNotSD6RBw==
|
integrity sha512-MqBkQh/OHTS2egovRtLk45wEyNXwF+cokD+1YPf9u5VfJiRdAiRwB2froX5Co9Rh20xs4siNPm8naNotSD6RBw==
|
||||||
|
Loading…
Reference in New Issue
Block a user