Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Added multichain api notifications #25869

Merged
merged 41 commits into from
Aug 19, 2024
Merged
Show file tree
Hide file tree
Changes from 8 commits
Commits
Show all changes
41 commits
Select commit Hold shift + click to select a range
70b2068
Added initial WIP
shanejonas Jul 16, 2024
9e8f0df
pushing WIP WIP
shanejonas Jul 16, 2024
e596389
fix: get it almost there
shanejonas Jul 16, 2024
02ebac6
lint
jiexi Jul 16, 2024
a1e4c9c
Merge branch 'caip-multichain' into sj/caip-multichain-notifications
jiexi Jul 16, 2024
d093eab
fix: issue with destroy
shanejonas Jul 17, 2024
f5fb9c7
fix: add tests for multichain subscription manage and middleware manager
shanejonas Jul 18, 2024
dc671f9
fix: wip
shanejonas Jul 23, 2024
cd6f10e
fix: wip
shanejonas Jul 23, 2024
c51e95f
fix: get destroy working and fix linting
shanejonas Jul 24, 2024
44e5f81
fix: remove console logs
shanejonas Jul 24, 2024
4ac28e5
Merge branch 'caip-multichain' into sj/caip-multichain-notifications
shanejonas Jul 24, 2024
52003d2
fix: remove hook
shanejonas Jul 24, 2024
12c15e2
fix: add nullish coalescing operating instead of ||
shanejonas Jul 29, 2024
adf042c
fix: NetworkController types
shanejonas Jul 29, 2024
c098ef4
fix: remove unused import
shanejonas Jul 29, 2024
2f39c90
fix: typings for onNotification + middleware destroy
shanejonas Jul 30, 2024
c547db3
fix: linting
shanejonas Jul 30, 2024
a5f73ec
fix: add notifications in stateChange instead of provider authorize
shanejonas Jul 30, 2024
7c48fab
Merge branch 'caip-multichain' into sj/caip-multichain-notifications
jiexi Jul 30, 2024
757a205
Fix
jiexi Jul 30, 2024
e126aab
Merge branch 'caip-multichain' into sj/caip-multichain-notifications
jiexi Jul 30, 2024
2e63b0f
fix: stateChange logic
shanejonas Jul 30, 2024
820abb9
fix: remove unused code
shanejonas Jul 30, 2024
aeee091
Merge branch 'caip-multichain' into sj/caip-multichain-notifications
jiexi Jul 30, 2024
581fe0c
fix: add more tests for MultichainMiddlewareManager and subscriptionM…
shanejonas Aug 1, 2024
a51fb95
fix: remove console log
shanejonas Aug 1, 2024
87c0017
fix: destroy
shanejonas Aug 1, 2024
b5cd7d9
use intersection type instead of union type for adding destroy to mid…
adonesky1 Aug 1, 2024
1a88f4e
fix: type for middleware
shanejonas Aug 2, 2024
5ceb061
fix: more typings
shanejonas Aug 2, 2024
0aa3d00
fix: linting
shanejonas Aug 2, 2024
3df485d
fix: remove unneeded typing
shanejonas Aug 2, 2024
2a5920c
Rename multichainSubscriptionManager.ts to MultichainSubscriptionMana…
shanejonas Aug 2, 2024
1147629
fix: convert to class and set middleware destroy in constructor
shanejonas Aug 2, 2024
f172d1b
fix: remove destroy typings from test
shanejonas Aug 2, 2024
77760fe
fix: add note about JsonRpcMiddleware type and give more descriptive …
shanejonas Aug 19, 2024
39db51e
fix: add tests and feedback for removed authorizations selector
shanejonas Aug 19, 2024
c22c29e
Merge branch 'caip-multichain' into sj/caip-multichain-notifications
shanejonas Aug 19, 2024
804a55b
fix: dedupe
shanejonas Aug 19, 2024
9b9b581
Update LavaMoat policies
metamaskbot Aug 19, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
shanejonas marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@
import { JsonRpcMiddleware } from 'json-rpc-engine';
import MultichainSubscriptionManager, { createMultichainMiddlewareManager } from './multichainSubscriptionManager';

const newHeadsNotificationMock = {
method: 'eth_subscription',
params: {
result: {
difficulty: '0x15d9223a23aa',
extraData: '0xd983010305844765746887676f312e342e328777696e646f7773',
gasLimit: '0x47e7c4',
gasUsed: '0x38658',
logsBloom:
'0x00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000',
miner: '0xf8b483dba2c3b7176a3da549ad41a48bb3121069',
nonce: '0x084149998194cc5f',
number: '0x1348c9',
parentHash:
'0x7736fab79e05dc611604d22470dadad26f56fe494421b5b333de816ce1f25701',
receiptRoot:
'0x2fab35823ad00c7bb388595cb46652fe7886e00660a01e867824d3dceb1c8d36',
sha3Uncles:
'0x1dcc4de8dec75d7aab85b567b6ccd41ad312451b948a7413f0a142fd40d49347',
stateRoot:
'0xb3346685172db67de536d8765c43c31009d0eb3bd9c501c9be3229203f15f378',
timestamp: '0x56ffeff8',
},
},
};

describe('multichainSubscriptionManager', () => {
it('should subscribe to a chain', (done) => {
const domain = 'example.com';
const scope = 'eip155:1';
const mockFindNetworkClientIdByChainId = jest.fn()
const mockGetNetworkClientById = jest.fn().mockImplementation(() => ({
blockTracker: {},
provider: {},
}));
const subscriptionManager = new MultichainSubscriptionManager({
findNetworkClientIdByChainId: mockFindNetworkClientIdByChainId,
getNetworkClientById: mockGetNetworkClientById,
});
subscriptionManager.subscribe(scope, domain);
subscriptionManager.on('notification', (domain: string, notification: any) => {
expect(notification).toMatchObject({
method: "wallet_invokeMethod",
params: {
scope,
request: newHeadsNotificationMock,
}
});
done();
});
subscriptionManager.onNotification(scope, domain, newHeadsNotificationMock);
});
});
describe('multichainMiddlewareManager', () => {
it('should add middleware and get called for the scope', () => {
const multichainMiddlewareManager = createMultichainMiddlewareManager();
const middlewareSpy = jest.fn();
const domain = 'example.com';
multichainMiddlewareManager.addMiddleware('eip155:1', domain, middlewareSpy);
multichainMiddlewareManager.middleware({scope: 'eip155:1'}, {}, () => { }, () => {});
expect(middlewareSpy).toHaveBeenCalled();
})
it('should remove middleware', () => {
const multichainMiddlewareManager = createMultichainMiddlewareManager();
const middlewareMock = jest.fn();
(middlewareMock as unknown as {destroy: () => void}).destroy = jest.fn();
const scope = 'eip155:1';
const domain = 'example.com';
multichainMiddlewareManager.addMiddleware(scope, domain, middlewareMock);
multichainMiddlewareManager.removeMiddleware(scope);
const nextSpy = jest.fn();
multichainMiddlewareManager.middleware({scope}, {}, nextSpy, () => {});
expect(nextSpy).toHaveBeenCalled();
})
});
194 changes: 194 additions & 0 deletions app/scripts/lib/multichain-api/multichainSubscriptionManager.ts
shanejonas marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,194 @@
import {
NetworkControllerFindNetworkClientIdByChainIdAction,
NetworkControllerGetNetworkClientByIdAction,
} from '@metamask/network-controller';
import SafeEventEmitter from '@metamask/safe-event-emitter';
import { parseCaipChainId } from '@metamask/utils';
import { toHex } from '@metamask/controller-utils';
import { Scope } from './scope';
import { JsonRpcMiddleware } from 'json-rpc-engine';

const createSubscriptionManager = require('@metamask/eth-json-rpc-filters/subscriptionManager');
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

what's the issue with using import? types?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yea theres no types available for this package


type MultichainSubscriptionManagerOptions = {
findNetworkClientIdByChainId: NetworkControllerFindNetworkClientIdByChainIdAction['handler'];
shanejonas marked this conversation as resolved.
Show resolved Hide resolved
getNetworkClientById: NetworkControllerGetNetworkClientByIdAction['handler'];
};

export default class MultichainSubscriptionManager extends SafeEventEmitter {
private subscriptionsByChain: {
[scope: string]: {
[domain: string]: unknown;
};
};

private findNetworkClientIdByChainId: NetworkControllerFindNetworkClientIdByChainIdAction['handler'];

private getNetworkClientById: NetworkControllerGetNetworkClientByIdAction['handler'];
shanejonas marked this conversation as resolved.
Show resolved Hide resolved

private subscriptionManagerByChain: { [scope: string]: any };
private subscriptionsCountByScope: { [scope: string]: number };
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What is the subscriptionsCountByScope for?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

oh nvm I see for destroying the manager when it hits zero

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

thoughts on using Object.keys(this.subscriptionsByChain[scope]).length instead?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it needs to be not the scope length but how many domains we have going with subscriptions, maybe it should be renamed


constructor(options: MultichainSubscriptionManagerOptions) {
super();
this.findNetworkClientIdByChainId = options.findNetworkClientIdByChainId;
this.getNetworkClientById = options.getNetworkClientById;
this.subscriptionManagerByChain = {};
this.subscriptionsByChain = {};
this.subscriptionsCountByScope = {};
}

onNotification(scope: Scope, domain: string, message: any) {
this.emit('notification', domain, {
method: 'wallet_invokeMethod',
params: {
scope,
request: message,
},
});
}

subscribe(scope: Scope, domain: string) {
this.subscriptionsCountByScope[scope] =
this.subscriptionsCountByScope[scope] || 0;
this.subscriptionsCountByScope[scope]++;
let subscriptionManager;
if (this.subscriptionManagerByChain[scope]) {
subscriptionManager = this.subscriptionManagerByChain[scope];
} else {
const networkClientId = this.findNetworkClientIdByChainId(
toHex(parseCaipChainId(scope).reference),
);
const networkClient = this.getNetworkClientById(networkClientId);
subscriptionManager = createSubscriptionManager({
blockTracker: networkClient.blockTracker,
provider: networkClient.provider,
});
this.subscriptionManagerByChain[scope] = subscriptionManager;
shanejonas marked this conversation as resolved.
Show resolved Hide resolved
}
this.subscriptionsByChain[scope] = this.subscriptionsByChain[scope] || {};
this.subscriptionsByChain[scope][domain] = this.onNotification.bind(
this,
scope,
domain,
);
subscriptionManager.events.on(
'notification',
this.subscriptionsByChain[scope][domain],
);
return subscriptionManager;
}

unsubscribe(scope: Scope, domain: string) {
const subscriptionManager = this.subscriptionManagerByChain[scope];
if (subscriptionManager && this.subscriptionsByChain[scope][domain]) {
subscriptionManager.events.off(
'notification',
this.subscriptionsByChain[scope][domain],
);
delete this.subscriptionsByChain[scope][domain];
}
if (this.subscriptionsCountByScope[scope]) {
this.subscriptionsCountByScope[scope]--;
if (this.subscriptionsCountByScope[scope] === 0) {
subscriptionManager.destroy();
delete this.subscriptionsCountByScope[scope];
delete this.subscriptionManagerByChain[scope];
delete this.subscriptionsByChain[scope];
}
}
}

unsubscribeAll() {
Object.entries(this.subscriptionsByChain).forEach(
([scope, domainObject]) => {
Object.entries(domainObject).forEach(([domain]) => {
this.unsubscribe(scope, domain);
});
},
);
}
unsubscribeScope(scope: string) {
Object.entries(this.subscriptionsByChain).forEach(
([_scope, domainObject]) => {
if (scope === _scope) {
Object.entries(domainObject).forEach(([_domain]) => {
this.unsubscribe(_scope, _domain);
});
}
},
);
}

unsubscribeDomain(domain: string) {
Object.entries(this.subscriptionsByChain).forEach(
([scope, domainObject]) => {
Object.entries(domainObject).forEach(([_domain]) => {
if (domain === _domain) {
this.unsubscribe(scope, _domain);
}
shanejonas marked this conversation as resolved.
Show resolved Hide resolved
});
},
);
}
}

// per scope middleware to handle legacy middleware
export const createMultichainMiddlewareManager = () => {
const middlewaresByScope: Record<Scope, any> = {};
const middlewareCountByDomainAndScope: Record<Scope, Record<string, number>> = {};
const removeMiddleware = (scope: Scope, domain?: string) => {
middlewareCountByDomainAndScope[scope] = middlewareCountByDomainAndScope[scope] || {};
if (domain) {
middlewareCountByDomainAndScope[scope][domain]--;
}
if (typeof domain === 'undefined' || middlewareCountByDomainAndScope[scope][domain] <= 0) {
const middleware = middlewaresByScope[scope];
if (domain) {
delete middlewareCountByDomainAndScope[scope][domain];
}
middleware.destroy();
delete middlewaresByScope[scope];
}
};

const removeAllMiddlewareForDomain = (domain: string) => {
for (const [scope, domains] of Object.entries(middlewareCountByDomainAndScope)) {
for (const [_domain] of Object.entries(domains)) {
if (_domain === domain) {
removeMiddleware(scope, domain);
}
}
}
};

const removeAllMiddleware = () => {
for (const [scope, domainObject] of Object.entries(middlewareCountByDomainAndScope)) {
for (const domain of Object.keys(domainObject)) {
removeMiddleware(scope, domain);
}
}
};

const addMiddleware = (scope: Scope, domain: string, middleware: JsonRpcMiddleware<unknown, unknown>) => {
middlewareCountByDomainAndScope[scope] = middlewareCountByDomainAndScope[scope] || {};
middlewareCountByDomainAndScope[scope][domain] = middlewareCountByDomainAndScope[scope][domain] || 0;
middlewareCountByDomainAndScope[scope][domain]++;
if (!middlewaresByScope[scope]) {
middlewaresByScope[scope] = middleware;
}
};

return {
middleware: (req: any, res: any, next: any, end: any) => {
if (!middlewaresByScope[req.scope]) {
return next();
}
return middlewaresByScope[req.scope](req, res, next, end);
},
addMiddleware,
removeMiddleware,
removeAllMiddleware,
removeAllMiddlewareForDomain
};
};
32 changes: 27 additions & 5 deletions app/scripts/lib/multichain-api/provider-authorize.js
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,6 @@ export async function providerAuthorizeHandler(req, res, _next, end, hooks) {
),
);
}

const sessionId = '0xdeadbeef';

if (sessionProperties && Object.keys(sessionProperties).length === 0) {
Expand All @@ -72,6 +71,7 @@ export async function providerAuthorizeHandler(req, res, _next, end, hooks) {
[RestrictedMethods.eth_accounts]: {},
},
);
console.log(' got to after requestPermissions');
shanejonas marked this conversation as resolved.
Show resolved Hide resolved
const permittedAccounts = getAccountsFromPermission(subjectPermission);
const { flattenedRequiredScopes, flattenedOptionalScopes } = processScopes(
requiredScopes,
Expand Down Expand Up @@ -113,14 +113,36 @@ export async function providerAuthorizeHandler(req, res, _next, end, hooks) {
},
});

const mergedScopes = mergeScopes(
flattenedRequiredScopes,
flattenedOptionalScopes,
);

hooks.multichainMiddlewareManager.removeAllMiddlewareForDomain(origin);
hooks.multichainSubscriptionManager.unsubscribeDomain(origin);
shanejonas marked this conversation as resolved.
Show resolved Hide resolved

// if the eth_subscription notification is in the scope and eth_subscribe is in the methods
// then get the subscriptionManager going for that scope
Object.entries(mergedScopes).forEach(([scope, scopeObject]) => {
shanejonas marked this conversation as resolved.
Show resolved Hide resolved
if (
scopeObject.notifications.includes('eth_subscription') &&
shanejonas marked this conversation as resolved.
Show resolved Hide resolved
scopeObject.methods.includes('eth_subscribe')
) {
const subscriptionManager =
hooks.multichainSubscriptionManager.subscribe(scope, origin);
hooks.multichainMiddlewareManager.addMiddleware(
scope,
origin,
subscriptionManager.middleware,
);
}
});

// TODO: metrics/tracking after approval

res.result = {
sessionId,
sessionScopes: mergeScopes(
flattenedRequiredScopes,
flattenedOptionalScopes,
),
sessionScopes: mergedScopes,
sessionProperties,
};
return end();
Expand Down
15 changes: 15 additions & 0 deletions app/scripts/lib/multichain-api/provider-authorize.test.js
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

should add scenarios to cover this new feature

Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import {
Caip25CaveatType,
Caip25EndowmentPermissionName,
} from './caip25permissions';
import { unsubscribe } from 'diagnostics_channel';
shanejonas marked this conversation as resolved.
Show resolved Hide resolved

jest.mock('./scope', () => ({
...jest.requireActual('./scope'),
Expand Down Expand Up @@ -58,6 +59,19 @@ const createMockedHandler = () => {
const response = {};
const handler = (request) =>
providerAuthorizeHandler(request, response, next, end, {
multichainMiddlewareManager: {
addMiddleware: jest.fn(),
removeMiddleware: jest.fn(),
removeAllMiddleware: jest.fn(),
removeAllMiddlewareForDomain: jest.fn(),
},
multichainSubscriptionManager: {
subscribe: jest.fn(),
unsubscribe: jest.fn(),
unsubscribeAll: jest.fn(),
unsubscribeDomain: jest.fn(),
unsubscribeScope: jest.fn(),
},
findNetworkClientIdByChainId,
requestPermissions,
grantPermissions,
Expand Down Expand Up @@ -295,6 +309,7 @@ describe('provider_authorize', () => {
},
});
await handler(baseRequest);
console.log('RESPONSE ===', response);
shanejonas marked this conversation as resolved.
Show resolved Hide resolved

expect(response.result).toStrictEqual({
sessionId: '0xdeadbeef',
Expand Down
1 change: 1 addition & 0 deletions app/scripts/lib/multichain-api/provider-request.js
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,7 @@ export async function providerRequestHandler(
}

Object.assign(request, {
scope,
networkClientId,
method: wrappedRequest.method,
params: wrappedRequest.params,
Expand Down
2 changes: 2 additions & 0 deletions app/scripts/lib/multichain-api/provider-request.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -154,6 +154,7 @@ describe('provider_request', () => {

await handler(request);
expect(request).toStrictEqual({
scope: 'eip155:1',
origin: 'http://test.com',
networkClientId: 'mainnet',
method: 'eth_call',
Expand Down Expand Up @@ -223,6 +224,7 @@ describe('provider_request', () => {
};
await handler(walletRequest);
expect(walletRequest).toStrictEqual({
scope: 'wallet',
origin: 'http://test.com',
networkClientId: 'selectedNetworkClientId',
method: 'wallet_watchAsset',
Expand Down
4 changes: 3 additions & 1 deletion app/scripts/lib/multichain-api/scope/supported.ts
Original file line number Diff line number Diff line change
Expand Up @@ -80,5 +80,7 @@ export const isSupportedAccount = (
// TODO: Needs to go into a capabilties/routing controller
// TODO: These make no sense in a multichain world. accountsChange becomes authorization/permissionChanged?
export const isSupportedNotification = (notification: string): boolean => {
return ['accountsChanged', 'chainChanged'].includes(notification);
return ['accountsChanged', 'chainChanged', 'eth_subscription'].includes(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we may not want either accountsChanged but definitely not chainChanged supported over this API

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the CAIP specs should be updated to not show these notifications as examples anymore

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

please update the spec for this function when you get a chance 🙏

notification,
);
};
Loading