@metamask/snaps-jest 8.0.0 → 8.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (80) hide show
  1. package/CHANGELOG.md +16 -1
  2. package/dist/{chunk-573UCCBF.js → chunk-2OHD2VKS.js} +72 -20
  3. package/dist/chunk-2OHD2VKS.js.map +1 -0
  4. package/dist/{chunk-A3EYYEAY.js → chunk-3FDEYMQU.js} +7 -7
  5. package/dist/{chunk-DTDYOBCI.js → chunk-44ARQBXS.js} +3 -3
  6. package/dist/{chunk-QSIY5NAK.mjs → chunk-6V5GEUDO.mjs} +2 -2
  7. package/dist/{chunk-QSIY5NAK.mjs.map → chunk-6V5GEUDO.mjs.map} +1 -1
  8. package/dist/{chunk-B23SV5TL.mjs → chunk-AFA4KKWW.mjs} +95 -18
  9. package/dist/chunk-AFA4KKWW.mjs.map +1 -0
  10. package/dist/{chunk-VSUNWJMQ.mjs → chunk-E2BYTLOT.mjs} +3 -3
  11. package/dist/{chunk-MCYTJUDG.js → chunk-EDFQDZNY.js} +6 -6
  12. package/dist/{chunk-WH5C5WIZ.mjs → chunk-FQWOVTBB.mjs} +4 -3
  13. package/dist/chunk-FQWOVTBB.mjs.map +1 -0
  14. package/dist/{chunk-LG3VKXGH.mjs → chunk-GG2BCPQH.mjs} +2 -2
  15. package/dist/{chunk-MJHR5RTY.js → chunk-J3I5KZIF.js} +4 -3
  16. package/dist/chunk-J3I5KZIF.js.map +1 -0
  17. package/dist/{chunk-2JAGD4N6.js → chunk-JXAJWUVZ.js} +3 -3
  18. package/dist/{chunk-X5IPMTHO.mjs → chunk-K55LIE3W.mjs} +2 -2
  19. package/dist/{chunk-2SBUXBSN.js → chunk-KQBJQLZG.js} +2 -2
  20. package/dist/chunk-KQBJQLZG.js.map +1 -0
  21. package/dist/{chunk-ZWN4SO2V.js → chunk-MBQHVBLA.js} +4 -4
  22. package/dist/{chunk-IS4GY5A2.js → chunk-R55KAAM4.js} +95 -18
  23. package/dist/chunk-R55KAAM4.js.map +1 -0
  24. package/dist/{chunk-TAIJXTLU.mjs → chunk-TMAWGVZP.mjs} +2 -2
  25. package/dist/{chunk-R7PQRB54.mjs → chunk-U3DVRTS2.mjs} +2 -2
  26. package/dist/{chunk-QC3BL5R2.mjs → chunk-UPR3PTSA.mjs} +70 -18
  27. package/dist/chunk-UPR3PTSA.mjs.map +1 -0
  28. package/dist/environment.js +8 -8
  29. package/dist/environment.mjs +7 -7
  30. package/dist/helpers.js +8 -8
  31. package/dist/helpers.mjs +7 -7
  32. package/dist/index.js +9 -9
  33. package/dist/index.mjs +8 -8
  34. package/dist/internals/index.js +9 -7
  35. package/dist/internals/index.mjs +8 -6
  36. package/dist/internals/request.js +8 -7
  37. package/dist/internals/request.mjs +7 -6
  38. package/dist/internals/simulation/controllers.js +4 -4
  39. package/dist/internals/simulation/controllers.mjs +3 -3
  40. package/dist/internals/simulation/index.js +8 -6
  41. package/dist/internals/simulation/index.mjs +7 -5
  42. package/dist/internals/simulation/interface.js +4 -2
  43. package/dist/internals/simulation/interface.mjs +3 -1
  44. package/dist/internals/simulation/methods/hooks/index.js +2 -2
  45. package/dist/internals/simulation/methods/hooks/index.mjs +1 -1
  46. package/dist/internals/simulation/methods/hooks/interface.js +2 -2
  47. package/dist/internals/simulation/methods/hooks/interface.mjs +1 -1
  48. package/dist/internals/simulation/methods/index.js +3 -3
  49. package/dist/internals/simulation/methods/index.mjs +2 -2
  50. package/dist/internals/simulation/methods/specifications.js +3 -3
  51. package/dist/internals/simulation/methods/specifications.mjs +2 -2
  52. package/dist/internals/simulation/simulation.js +5 -5
  53. package/dist/internals/simulation/simulation.mjs +4 -4
  54. package/dist/matchers.js +8 -8
  55. package/dist/matchers.mjs +7 -7
  56. package/dist/setup.js +7 -7
  57. package/dist/setup.mjs +7 -7
  58. package/dist/tsconfig.build.tsbuildinfo +1 -1
  59. package/dist/types/internals/request.d.ts +1 -1
  60. package/dist/types/internals/simulation/interface.d.ts +11 -0
  61. package/dist/types/internals/simulation/methods/hooks/interface.d.ts +2 -2
  62. package/dist/types/types.d.ts +7 -0
  63. package/package.json +9 -9
  64. package/dist/chunk-2SBUXBSN.js.map +0 -1
  65. package/dist/chunk-573UCCBF.js.map +0 -1
  66. package/dist/chunk-B23SV5TL.mjs.map +0 -1
  67. package/dist/chunk-IS4GY5A2.js.map +0 -1
  68. package/dist/chunk-MJHR5RTY.js.map +0 -1
  69. package/dist/chunk-QC3BL5R2.mjs.map +0 -1
  70. package/dist/chunk-WH5C5WIZ.mjs.map +0 -1
  71. /package/dist/{chunk-A3EYYEAY.js.map → chunk-3FDEYMQU.js.map} +0 -0
  72. /package/dist/{chunk-DTDYOBCI.js.map → chunk-44ARQBXS.js.map} +0 -0
  73. /package/dist/{chunk-VSUNWJMQ.mjs.map → chunk-E2BYTLOT.mjs.map} +0 -0
  74. /package/dist/{chunk-MCYTJUDG.js.map → chunk-EDFQDZNY.js.map} +0 -0
  75. /package/dist/{chunk-LG3VKXGH.mjs.map → chunk-GG2BCPQH.mjs.map} +0 -0
  76. /package/dist/{chunk-2JAGD4N6.js.map → chunk-JXAJWUVZ.js.map} +0 -0
  77. /package/dist/{chunk-X5IPMTHO.mjs.map → chunk-K55LIE3W.mjs.map} +0 -0
  78. /package/dist/{chunk-ZWN4SO2V.js.map → chunk-MBQHVBLA.js.map} +0 -0
  79. /package/dist/{chunk-TAIJXTLU.mjs.map → chunk-TMAWGVZP.mjs.map} +0 -0
  80. /package/dist/{chunk-R7PQRB54.mjs.map → chunk-U3DVRTS2.mjs.map} +0 -0
@@ -10,10 +10,10 @@ var _chunkVXPCUDBIjs = require('./chunk-VXPCUDBI.js');
10
10
 
11
11
 
12
12
 
13
- var _chunkDTDYOBCIjs = require('./chunk-DTDYOBCI.js');
13
+ var _chunk44ARQBXSjs = require('./chunk-44ARQBXS.js');
14
14
 
15
15
 
16
- var _chunkZWN4SO2Vjs = require('./chunk-ZWN4SO2V.js');
16
+ var _chunkMBQHVBLAjs = require('./chunk-MBQHVBLA.js');
17
17
 
18
18
 
19
19
  var _chunkWMLSD6B5js = require('./chunk-WMLSD6B5.js');
@@ -44,7 +44,7 @@ async function handleInstallSnap(snapId, {
44
44
  const controllerMessenger = new (0, _basecontroller.ControllerMessenger)();
45
45
  registerActions(controllerMessenger);
46
46
  const hooks = getHooks(options, snapFiles, snapId, controllerMessenger);
47
- const { subjectMetadataController, permissionController } = _chunkDTDYOBCIjs.getControllers.call(void 0, {
47
+ const { subjectMetadataController, permissionController } = _chunk44ARQBXSjs.getControllers.call(void 0, {
48
48
  controllerMessenger,
49
49
  hooks,
50
50
  runSaga,
@@ -76,14 +76,14 @@ async function handleInstallSnap(snapId, {
76
76
  });
77
77
  }
78
78
  });
79
- await _chunkDTDYOBCIjs.registerSnap.call(void 0, snapId, snapFiles.manifest.result, {
79
+ await _chunk44ARQBXSjs.registerSnap.call(void 0, snapId, snapFiles.manifest.result, {
80
80
  permissionController,
81
81
  subjectMetadataController
82
82
  });
83
83
  await service.executeSnap({
84
84
  snapId,
85
85
  sourceCode: snapFiles.sourceCode.toString("utf8"),
86
- endowments: await _chunkZWN4SO2Vjs.getEndowments.call(void 0, permissionController, snapId)
86
+ endowments: await _chunkMBQHVBLAjs.getEndowments.call(void 0, permissionController, snapId)
87
87
  });
88
88
  return {
89
89
  snapId,
@@ -131,4 +131,4 @@ function registerActions(controllerMessenger) {
131
131
 
132
132
 
133
133
  exports.handleInstallSnap = handleInstallSnap; exports.getHooks = getHooks; exports.registerActions = registerActions;
134
- //# sourceMappingURL=chunk-MCYTJUDG.js.map
134
+ //# sourceMappingURL=chunk-EDFQDZNY.js.map
@@ -1,9 +1,10 @@
1
1
  // src/internals/simulation/methods/hooks/interface.ts
2
2
  function getCreateInterfaceImplementation(controllerMessenger) {
3
- return async (snapId, content) => controllerMessenger.call(
3
+ return async (snapId, content, context) => controllerMessenger.call(
4
4
  "SnapInterfaceController:createInterface",
5
5
  snapId,
6
- content
6
+ content,
7
+ context
7
8
  );
8
9
  }
9
10
  function getGetInterfaceImplementation(controllerMessenger) {
@@ -18,4 +19,4 @@ export {
18
19
  getCreateInterfaceImplementation,
19
20
  getGetInterfaceImplementation
20
21
  };
21
- //# sourceMappingURL=chunk-WH5C5WIZ.mjs.map
22
+ //# sourceMappingURL=chunk-FQWOVTBB.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/internals/simulation/methods/hooks/interface.ts"],"sourcesContent":["import type { Component, InterfaceContext, SnapId } from '@metamask/snaps-sdk';\n\nimport type { RootControllerMessenger } from '../../controllers';\n\n/**\n * Get the implementation of the `createInterface` hook.\n *\n * @param controllerMessenger - The controller messenger used to call actions.\n * @returns The implementation of the `createInterface` hook.\n */\nexport function getCreateInterfaceImplementation(\n controllerMessenger: RootControllerMessenger,\n) {\n return async (\n snapId: SnapId,\n content: Component,\n context?: InterfaceContext,\n ) =>\n controllerMessenger.call(\n 'SnapInterfaceController:createInterface',\n snapId,\n content,\n context,\n );\n}\n\n/**\n * Get the implementation of the `getInterface` hook.\n *\n * @param controllerMessenger - The controller messenger used to call actions.\n * @returns The implementation of the `getInterface` hook.\n */\nexport function getGetInterfaceImplementation(\n controllerMessenger: RootControllerMessenger,\n) {\n return (snapId: SnapId, id: string) =>\n controllerMessenger.call(\n 'SnapInterfaceController:getInterface',\n snapId,\n id,\n );\n}\n"],"mappings":";AAUO,SAAS,iCACd,qBACA;AACA,SAAO,OACL,QACA,SACA,YAEA,oBAAoB;AAAA,IAClB;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF;AACJ;AAQO,SAAS,8BACd,qBACA;AACA,SAAO,CAAC,QAAgB,OACtB,oBAAoB;AAAA,IAClB;AAAA,IACA;AAAA,IACA;AAAA,EACF;AACJ;","names":[]}
@@ -1,6 +1,6 @@
1
1
  import {
2
2
  getPermissionSpecifications
3
- } from "./chunk-TAIJXTLU.mjs";
3
+ } from "./chunk-TMAWGVZP.mjs";
4
4
  import {
5
5
  UNRESTRICTED_METHODS
6
6
  } from "./chunk-57SGDM5B.mjs";
@@ -92,4 +92,4 @@ export {
92
92
  getControllers,
93
93
  registerSnap
94
94
  };
95
- //# sourceMappingURL=chunk-LG3VKXGH.mjs.map
95
+ //# sourceMappingURL=chunk-GG2BCPQH.mjs.map
@@ -1,9 +1,10 @@
1
1
  "use strict";Object.defineProperty(exports, "__esModule", {value: true});// src/internals/simulation/methods/hooks/interface.ts
2
2
  function getCreateInterfaceImplementation(controllerMessenger) {
3
- return async (snapId, content) => controllerMessenger.call(
3
+ return async (snapId, content, context) => controllerMessenger.call(
4
4
  "SnapInterfaceController:createInterface",
5
5
  snapId,
6
- content
6
+ content,
7
+ context
7
8
  );
8
9
  }
9
10
  function getGetInterfaceImplementation(controllerMessenger) {
@@ -18,4 +19,4 @@ function getGetInterfaceImplementation(controllerMessenger) {
18
19
 
19
20
 
20
21
  exports.getCreateInterfaceImplementation = getCreateInterfaceImplementation; exports.getGetInterfaceImplementation = getGetInterfaceImplementation;
21
- //# sourceMappingURL=chunk-MJHR5RTY.js.map
22
+ //# sourceMappingURL=chunk-J3I5KZIF.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/internals/simulation/methods/hooks/interface.ts"],"names":[],"mappings":";AAUO,SAAS,iCACd,qBACA;AACA,SAAO,OACL,QACA,SACA,YAEA,oBAAoB;AAAA,IAClB;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF;AACJ;AAQO,SAAS,8BACd,qBACA;AACA,SAAO,CAAC,QAAgB,OACtB,oBAAoB;AAAA,IAClB;AAAA,IACA;AAAA,IACA;AAAA,EACF;AACJ","sourcesContent":["import type { Component, InterfaceContext, SnapId } from '@metamask/snaps-sdk';\n\nimport type { RootControllerMessenger } from '../../controllers';\n\n/**\n * Get the implementation of the `createInterface` hook.\n *\n * @param controllerMessenger - The controller messenger used to call actions.\n * @returns The implementation of the `createInterface` hook.\n */\nexport function getCreateInterfaceImplementation(\n controllerMessenger: RootControllerMessenger,\n) {\n return async (\n snapId: SnapId,\n content: Component,\n context?: InterfaceContext,\n ) =>\n controllerMessenger.call(\n 'SnapInterfaceController:createInterface',\n snapId,\n content,\n context,\n );\n}\n\n/**\n * Get the implementation of the `getInterface` hook.\n *\n * @param controllerMessenger - The controller messenger used to call actions.\n * @returns The implementation of the `getInterface` hook.\n */\nexport function getGetInterfaceImplementation(\n controllerMessenger: RootControllerMessenger,\n) {\n return (snapId: SnapId, id: string) =>\n controllerMessenger.call(\n 'SnapInterfaceController:getInterface',\n snapId,\n id,\n );\n}\n"]}
@@ -3,7 +3,7 @@
3
3
  var _chunkT2PHS5FYjs = require('./chunk-T2PHS5FY.js');
4
4
 
5
5
 
6
- var _chunkMCYTJUDGjs = require('./chunk-MCYTJUDG.js');
6
+ var _chunkEDFQDZNYjs = require('./chunk-EDFQDZNY.js');
7
7
 
8
8
 
9
9
  var _chunkN6MAO223js = require('./chunk-N6MAO223.js');
@@ -72,7 +72,7 @@ var SnapsEnvironment = class extends _jestenvironmentnode2.default {
72
72
  */
73
73
  async installSnap(snapId = this.snapId, options = {}) {
74
74
  await _chunkPHUTP7NBjs.__privateGet.call(void 0, this, _instance)?.executionService.terminateAllSnaps();
75
- _chunkPHUTP7NBjs.__privateSet.call(void 0, this, _instance, await _chunkMCYTJUDGjs.handleInstallSnap.call(void 0, snapId, options));
75
+ _chunkPHUTP7NBjs.__privateSet.call(void 0, this, _instance, await _chunkEDFQDZNYjs.handleInstallSnap.call(void 0, snapId, options));
76
76
  return _chunkPHUTP7NBjs.__privateGet.call(void 0, this, _instance);
77
77
  }
78
78
  /**
@@ -101,4 +101,4 @@ var environment_default = SnapsEnvironment;
101
101
 
102
102
 
103
103
  exports.SnapsEnvironment = SnapsEnvironment; exports.environment_default = environment_default;
104
- //# sourceMappingURL=chunk-2JAGD4N6.js.map
104
+ //# sourceMappingURL=chunk-JXAJWUVZ.js.map
@@ -3,7 +3,7 @@ import {
3
3
  } from "./chunk-B73P64TE.mjs";
4
4
  import {
5
5
  handleInstallSnap
6
- } from "./chunk-VSUNWJMQ.mjs";
6
+ } from "./chunk-E2BYTLOT.mjs";
7
7
  import {
8
8
  startServer
9
9
  } from "./chunk-SLU4FNKX.mjs";
@@ -101,4 +101,4 @@ export {
101
101
  SnapsEnvironment,
102
102
  environment_default
103
103
  };
104
- //# sourceMappingURL=chunk-X5IPMTHO.mjs.map
104
+ //# sourceMappingURL=chunk-K55LIE3W.mjs.map
@@ -136,7 +136,7 @@ var toRenderLegacy = function(actual, expected) {
136
136
  const { content } = actual;
137
137
  const expectedElement = _snapsutils.getJsxElementFromComponent.call(void 0, expected);
138
138
  const pass = this.equals(content, expectedElement);
139
- const difference = _jestmatcherutils.diff.call(void 0, actual, expectedElement);
139
+ const difference = _jestmatcherutils.diff.call(void 0, expectedElement, content);
140
140
  const message = pass ? () => `${this.utils.matcherHint(".not.toRender")}
141
141
 
142
142
  Expected:
@@ -208,4 +208,4 @@ _globals.expect.extend({
208
208
 
209
209
 
210
210
  exports.toRespondWith = toRespondWith; exports.toRespondWithError = toRespondWithError; exports.toSendNotification = toSendNotification; exports.serialiseJsx = serialiseJsx; exports.toRender = toRender;
211
- //# sourceMappingURL=chunk-2SBUXBSN.js.map
211
+ //# sourceMappingURL=chunk-KQBJQLZG.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/matchers.ts"],"names":["message"],"mappings":"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","sourcesContent":["/* eslint-disable no-invalid-this */\n\n// Note: Because this file imports from `@jest/globals`, it can only be used in\n// a Jest environment. This is why it's not exported from the index file.\n\nimport type { MatcherFunction } from '@jest/expect';\nimport { expect } from '@jest/globals';\nimport type {\n NotificationType,\n EnumToUnion,\n ComponentOrElement,\n Component,\n} from '@metamask/snaps-sdk';\nimport type { JSXElement, SnapNode } from '@metamask/snaps-sdk/jsx';\nimport { isJSXElementUnsafe } from '@metamask/snaps-sdk/jsx';\nimport { getJsxElementFromComponent } from '@metamask/snaps-utils';\nimport type { Json } from '@metamask/utils';\nimport { hasProperty } from '@metamask/utils';\nimport type { MatcherHintOptions } from 'jest-matcher-utils';\nimport {\n EXPECTED_COLOR,\n diff,\n matcherErrorMessage,\n matcherHint,\n printReceived,\n printWithType,\n RECEIVED_COLOR,\n} from 'jest-matcher-utils';\nimport { is } from 'superstruct';\n\nimport { InterfaceStruct, SnapResponseStruct } from './internals';\nimport type { SnapResponse } from './types';\n\n/**\n * Ensure that the actual value is a response from the `request` function.\n *\n * @param actual - The actual value.\n * @param matcherName - The name of the matcher.\n * @param options - The matcher options.\n */\nfunction assertActualIsSnapResponse(\n actual: unknown,\n matcherName: string,\n options?: MatcherHintOptions,\n): asserts actual is SnapResponse {\n if (!is(actual, SnapResponseStruct)) {\n throw new Error(\n matcherErrorMessage(\n matcherHint(matcherName, undefined, undefined, options),\n `${RECEIVED_COLOR(\n 'received',\n )} value must be a response from the \\`request\\` function`,\n printWithType('Received', actual, printReceived),\n ),\n );\n }\n}\n\n/**\n * Ensure that the actual value is a response from the `request` function, and\n * that it has a `ui` property.\n *\n * @param actual - The actual value.\n * @param matcherName - The name of the matcher.\n * @param options - The matcher options.\n */\nfunction assertHasInterface(\n actual: unknown,\n matcherName: string,\n options?: MatcherHintOptions,\n): asserts actual is { content: JSXElement } {\n if (!is(actual, InterfaceStruct) || !actual.content) {\n throw new Error(\n matcherErrorMessage(\n matcherHint(matcherName, undefined, undefined, options),\n `${RECEIVED_COLOR('received')} value must have a \\`content\\` property`,\n printWithType('Received', actual, printReceived),\n ),\n );\n }\n}\n\n/**\n * Check if a JSON-RPC response matches the expected value. This matcher is\n * intended to be used with the `expect` global.\n *\n * @param actual - The actual response.\n * @param expected - The expected response.\n * @returns The status and message.\n */\nexport const toRespondWith: MatcherFunction<[expected: Json]> = function (\n actual,\n expected,\n) {\n assertActualIsSnapResponse(actual, 'toRespondWith');\n\n const { response } = actual;\n if (hasProperty(response, 'error')) {\n const message = () =>\n `${this.utils.matcherHint('.toRespondWith')}\\n\\n` +\n `Expected response: ${this.utils.printExpected(expected)}\\n` +\n `Received error: ${this.utils.printReceived(response.error)}`;\n\n return { message, pass: false };\n }\n\n const pass = this.equals(response.result, expected);\n const message = pass\n ? () =>\n `${this.utils.matcherHint('.not.toRespondWith')}\\n\\n` +\n `Expected: ${this.utils.printExpected(expected)}\\n` +\n `Received: ${this.utils.printReceived(response.result)}`\n : () =>\n `${this.utils.matcherHint('.toRespondWith')}\\n\\n` +\n `Expected: ${this.utils.printExpected(expected)}\\n` +\n `Received: ${this.utils.printReceived(response.result)}`;\n\n return { message, pass };\n};\n\nexport const toRespondWithError: MatcherFunction<[expected: Json]> = function (\n actual,\n expected,\n) {\n assertActualIsSnapResponse(actual, 'toRespondWithError');\n\n const { response } = actual;\n if (hasProperty(response, 'result')) {\n const message = () =>\n `${this.utils.matcherHint('.toRespondWithError')}\\n\\n` +\n `Expected error: ${this.utils.printExpected(expected)}\\n` +\n `Received result: ${this.utils.printReceived(response.result)}`;\n\n return { message, pass: false };\n }\n\n const pass = this.equals(response.error, expected);\n const message = pass\n ? () =>\n `${this.utils.matcherHint('.not.toRespondWithError')}\\n\\n` +\n `Expected: ${this.utils.printExpected(expected)}\\n` +\n `Received: ${this.utils.printReceived(response.error)}`\n : () =>\n `${this.utils.matcherHint('.toRespondWithError')}\\n\\n` +\n `Expected: ${this.utils.printExpected(expected)}\\n` +\n `Received: ${this.utils.printReceived(response.error)}`;\n\n return { message, pass };\n};\n\n/**\n * Check if the snap sent a notification with the expected message. This matcher\n * is intended to be used with the `expect` global.\n *\n * @param actual - The actual response.\n * @param expected - The expected notification message.\n * @param type - The expected notification type.\n * @returns The status and message.\n */\nexport const toSendNotification: MatcherFunction<\n [expected: string, type?: EnumToUnion<NotificationType> | undefined]\n> = function (actual, expected, type) {\n assertActualIsSnapResponse(actual, 'toSendNotification');\n\n const { notifications } = actual;\n const pass = notifications.some(\n (notification) =>\n this.equals(notification.message, expected) &&\n (type === undefined || notification.type === type),\n );\n\n const message = pass\n ? () =>\n `${this.utils.matcherHint('.not.toSendNotification')}\\n\\n` +\n `Expected: ${this.utils.printExpected(expected)}\\n` +\n `Expected type: ${this.utils.printExpected(type)}\\n` +\n `Received: ${this.utils.printReceived(notifications)}`\n : () =>\n `${this.utils.matcherHint('.toSendNotification')}\\n\\n` +\n `Expected: ${this.utils.printExpected(expected)}\\n` +\n `Expected type: ${this.utils.printExpected(type)}\\n` +\n `Received: ${this.utils.printReceived(notifications)}`;\n\n return { message, pass };\n};\n\n/**\n * Serialise a JSX prop to a string.\n *\n * @param prop - The JSX prop.\n * @returns The serialised JSX prop.\n */\nfunction serialiseProp(prop: unknown): string {\n if (typeof prop === 'string') {\n return `\"${prop}\"`;\n }\n\n return `{${JSON.stringify(prop)}}`;\n}\n\n/**\n * Serialise JSX props to a string.\n *\n * @param props - The JSX props.\n * @returns The serialised JSX props.\n */\nfunction serialiseProps(props: Record<string, unknown>): string {\n return Object.entries(props)\n .filter(([key]) => key !== 'children')\n .sort(([a], [b]) => a.localeCompare(b))\n .map(([key, value]) => ` ${key}=${serialiseProp(value)}`)\n .join('');\n}\n\n/**\n * Serialise a JSX node to a string.\n *\n * @param node - The JSX node.\n * @param indentation - The indentation level. Defaults to `0`. This should not\n * be set by the caller, as it is used for recursion.\n * @returns The serialised JSX node.\n */\nexport function serialiseJsx(node: SnapNode, indentation = 0): string {\n if (Array.isArray(node)) {\n return node.map((child) => serialiseJsx(child, indentation)).join('');\n }\n\n const indent = ' '.repeat(indentation);\n if (typeof node === 'string') {\n return `${indent}${node}\\n`;\n }\n\n if (!node) {\n return '';\n }\n\n const { type, props } = node;\n const trailingNewline = indentation > 0 ? '\\n' : '';\n\n if (hasProperty(props, 'children')) {\n const children = serialiseJsx(props.children as SnapNode, indentation + 1);\n return `${indent}<${type}${serialiseProps(\n props,\n )}>\\n${children}${indent}</${type}>${trailingNewline}`;\n }\n\n return `${indent}<${type}${serialiseProps(props)} />${trailingNewline}`;\n}\n\nconst toRenderLegacy: MatcherFunction<[expected: Component]> = function (\n actual,\n expected,\n) {\n assertHasInterface(actual, 'toRender');\n\n const { content } = actual;\n const expectedElement = getJsxElementFromComponent(expected);\n const pass = this.equals(content, expectedElement);\n\n // This is typed as `string | null`, but in practice it's always a string.\n // The function only returns `null` if both the expected and actual values\n // are numbers, bigints, or booleans, which is never the case here.\n const difference = diff(expectedElement, content) as string;\n\n const message = pass\n ? () =>\n `${this.utils.matcherHint('.not.toRender')}\\n\\n` +\n `Expected:\\n${this.utils.printExpected(expectedElement)}\\n\\n` +\n `Received:\\n${this.utils.printReceived(content)}` +\n `\\n\\nDifference:\\n\\n${difference}`\n : () =>\n `${this.utils.matcherHint('.toRender')}\\n\\n` +\n `Expected:\\n${this.utils.printExpected(expectedElement)}\\n\\n` +\n `Received:\\n${this.utils.printReceived(content)}` +\n `\\n\\nDifference:\\n\\n${difference}`;\n\n return { message, pass };\n};\n\nexport const toRender: MatcherFunction<[expected: ComponentOrElement]> =\n function (actual, expected) {\n assertHasInterface(actual, 'toRender');\n\n if (!isJSXElementUnsafe(expected)) {\n return toRenderLegacy.call(this, actual, expected);\n }\n\n const { content } = actual;\n const pass = this.equals(content, expected);\n\n // This is typed as `string | null`, but in practice it's always a string.\n // The function only returns `null` if both the expected and actual values\n // are numbers, bigints, or booleans, which is never the case here.\n const difference = diff(\n serialiseJsx(expected),\n serialiseJsx(content),\n ) as string;\n\n const message = pass\n ? () =>\n `${this.utils.matcherHint('.not.toRender')}\\n\\n` +\n `Expected:\\n${EXPECTED_COLOR(serialiseJsx(expected))}\\n\\n` +\n `Received:\\n${RECEIVED_COLOR(serialiseJsx(content))}` +\n `\\n\\nDifference:\\n\\n${difference}`\n : () =>\n `${this.utils.matcherHint('.toRender')}\\n\\n` +\n `Expected:\\n${EXPECTED_COLOR(serialiseJsx(expected))}\\n\\n` +\n `Received:\\n${RECEIVED_COLOR(serialiseJsx(content))}` +\n `\\n\\nDifference:\\n\\n${difference}`;\n\n return { message, pass };\n };\n\nexpect.extend({\n toRespondWith,\n toRespondWithError,\n toSendNotification,\n toRender,\n});\n"]}
@@ -4,7 +4,7 @@ var _chunk265BMFM5js = require('./chunk-265BMFM5.js');
4
4
 
5
5
 
6
6
 
7
- var _chunkMJHR5RTYjs = require('./chunk-MJHR5RTY.js');
7
+ var _chunkJ3I5KZIFjs = require('./chunk-J3I5KZIF.js');
8
8
 
9
9
 
10
10
 
@@ -58,8 +58,8 @@ function getPermissionSpecifications({
58
58
  showInAppNotification: _chunkGMTKFAWOjs.getShowInAppNotificationImplementation.call(void 0, runSaga),
59
59
  showNativeNotification: _chunkGMTKFAWOjs.getShowNativeNotificationImplementation.call(void 0, runSaga),
60
60
  updateSnapState: _chunkSB5EPHE3js.getUpdateSnapStateMethodImplementation.call(void 0, runSaga),
61
- createInterface: _chunkMJHR5RTYjs.getCreateInterfaceImplementation.call(void 0, controllerMessenger),
62
- getInterface: _chunkMJHR5RTYjs.getGetInterfaceImplementation.call(void 0, controllerMessenger)
61
+ createInterface: _chunkJ3I5KZIFjs.getCreateInterfaceImplementation.call(void 0, controllerMessenger),
62
+ getInterface: _chunkJ3I5KZIFjs.getGetInterfaceImplementation.call(void 0, controllerMessenger)
63
63
  })
64
64
  };
65
65
  }
@@ -86,4 +86,4 @@ async function getEndowments(permissionController, snapId) {
86
86
 
87
87
 
88
88
  exports.resolve = resolve; exports.asyncResolve = asyncResolve; exports.getPermissionSpecifications = getPermissionSpecifications; exports.getEndowments = getEndowments;
89
- //# sourceMappingURL=chunk-ZWN4SO2V.js.map
89
+ //# sourceMappingURL=chunk-MBQHVBLA.js.map
@@ -6,6 +6,11 @@ var _chunkPZDTZGSIjs = require('./chunk-PZDTZGSI.js');
6
6
 
7
7
  // src/internals/simulation/interface.ts
8
8
  var _snapssdk = require('@metamask/snaps-sdk');
9
+
10
+
11
+
12
+
13
+
9
14
  var _snapsutils = require('@metamask/snaps-utils');
10
15
  var _utils = require('@metamask/utils');
11
16
  var _effects = require('redux-saga/effects');
@@ -101,7 +106,7 @@ function getElement(content, name) {
101
106
  return void 0;
102
107
  });
103
108
  }
104
- async function handleEvent(controllerMessenger, snapId, id, event) {
109
+ async function handleEvent(controllerMessenger, snapId, id, event, context) {
105
110
  try {
106
111
  await controllerMessenger.call(
107
112
  "ExecutionService:handleRpcRequest",
@@ -114,7 +119,8 @@ async function handleEvent(controllerMessenger, snapId, id, event) {
114
119
  method: " ",
115
120
  params: {
116
121
  event,
117
- id
122
+ id,
123
+ context
118
124
  }
119
125
  }
120
126
  }
@@ -134,21 +140,33 @@ async function clickElement(controllerMessenger, id, content, snapId, name) {
134
140
  result.element.type === "Button",
135
141
  `Expected an element of type "Button", but found "${result.element.type}".`
136
142
  );
137
- await handleEvent(controllerMessenger, snapId, id, {
138
- type: _snapssdk.UserInputEventType.ButtonClickEvent,
139
- name: result.element.props.name
140
- });
143
+ const { state, context } = controllerMessenger.call(
144
+ "SnapInterfaceController:getInterface",
145
+ snapId,
146
+ id
147
+ );
148
+ await handleEvent(
149
+ controllerMessenger,
150
+ snapId,
151
+ id,
152
+ {
153
+ type: _snapssdk.UserInputEventType.ButtonClickEvent,
154
+ name: result.element.props.name
155
+ },
156
+ context
157
+ );
141
158
  if (result.form && result.element.props.type === "submit") {
142
- const { state } = controllerMessenger.call(
143
- "SnapInterfaceController:getInterface",
159
+ await handleEvent(
160
+ controllerMessenger,
144
161
  snapId,
145
- id
162
+ id,
163
+ {
164
+ type: _snapssdk.UserInputEventType.FormSubmitEvent,
165
+ name: result.form,
166
+ value: state[result.form]
167
+ },
168
+ context
146
169
  );
147
- await handleEvent(controllerMessenger, snapId, id, {
148
- type: _snapssdk.UserInputEventType.FormSubmitEvent,
149
- name: result.form,
150
- value: state[result.form]
151
- });
152
170
  }
153
171
  }
154
172
  function mergeValue(state, name, value, form) {
@@ -173,7 +191,54 @@ async function typeInField(controllerMessenger, id, content, snapId, name, value
173
191
  result.element.type === "Input",
174
192
  `Expected an element of type "Input", but found "${result.element.type}".`
175
193
  );
176
- const { state } = controllerMessenger.call(
194
+ const { state, context } = controllerMessenger.call(
195
+ "SnapInterfaceController:getInterface",
196
+ snapId,
197
+ id
198
+ );
199
+ const newState = mergeValue(state, name, value, result.form);
200
+ controllerMessenger.call(
201
+ "SnapInterfaceController:updateInterfaceState",
202
+ id,
203
+ newState
204
+ );
205
+ await controllerMessenger.call("ExecutionService:handleRpcRequest", snapId, {
206
+ origin: "",
207
+ handler: _snapsutils.HandlerType.OnUserInput,
208
+ request: {
209
+ jsonrpc: "2.0",
210
+ method: " ",
211
+ params: {
212
+ event: {
213
+ type: _snapssdk.UserInputEventType.InputChangeEvent,
214
+ name: result.element.props.name,
215
+ value
216
+ },
217
+ id,
218
+ context
219
+ }
220
+ }
221
+ });
222
+ }
223
+ async function selectInDropdown(controllerMessenger, id, content, snapId, name, value) {
224
+ const result = getElement(content, name);
225
+ _snapssdk.assert.call(void 0,
226
+ result !== void 0,
227
+ `Could not find an element in the interface with the name "${name}".`
228
+ );
229
+ _snapssdk.assert.call(void 0,
230
+ result.element.type === "Dropdown",
231
+ `Expected an element of type "Dropdown", but found "${result.element.type}".`
232
+ );
233
+ const options = _snapsutils.getJsxChildren.call(void 0, result.element);
234
+ const selectedOption = options.find(
235
+ (option) => _utils.hasProperty.call(void 0, option.props, "value") && option.props.value === value
236
+ );
237
+ _snapssdk.assert.call(void 0,
238
+ selectedOption !== void 0,
239
+ `The dropdown with the name "${name}" does not contain "${value}".`
240
+ );
241
+ const { state, context } = controllerMessenger.call(
177
242
  "SnapInterfaceController:getInterface",
178
243
  snapId,
179
244
  id
@@ -196,7 +261,8 @@ async function typeInField(controllerMessenger, id, content, snapId, name, value
196
261
  name: result.element.props.name,
197
262
  value
198
263
  },
199
- id
264
+ id,
265
+ context
200
266
  }
201
267
  }
202
268
  });
@@ -213,6 +279,16 @@ function* getInterface(runSaga, snapId, controllerMessenger) {
213
279
  },
214
280
  typeInField: async (name, value) => {
215
281
  await typeInField(controllerMessenger, id, content, snapId, name, value);
282
+ },
283
+ selectInDropdown: async (name, value) => {
284
+ await selectInDropdown(
285
+ controllerMessenger,
286
+ id,
287
+ content,
288
+ snapId,
289
+ name,
290
+ value
291
+ );
216
292
  }
217
293
  };
218
294
  return getInterfaceResponse(runSaga, type, content, interfaceActions);
@@ -225,5 +301,6 @@ function* getInterface(runSaga, snapId, controllerMessenger) {
225
301
 
226
302
 
227
303
 
228
- exports.getInterfaceResponse = getInterfaceResponse; exports.getElement = getElement; exports.clickElement = clickElement; exports.mergeValue = mergeValue; exports.typeInField = typeInField; exports.getInterface = getInterface;
229
- //# sourceMappingURL=chunk-IS4GY5A2.js.map
304
+
305
+ exports.getInterfaceResponse = getInterfaceResponse; exports.getElement = getElement; exports.clickElement = clickElement; exports.mergeValue = mergeValue; exports.typeInField = typeInField; exports.selectInDropdown = selectInDropdown; exports.getInterface = getInterface;
306
+ //# sourceMappingURL=chunk-R55KAAM4.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/internals/simulation/interface.ts"],"names":["content"],"mappings":"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","sourcesContent":["import type {\n FormState,\n InterfaceContext,\n InterfaceState,\n SnapId,\n UserInputEvent,\n} from '@metamask/snaps-sdk';\nimport { DialogType, UserInputEventType, assert } from '@metamask/snaps-sdk';\nimport type { FormElement, JSXElement } from '@metamask/snaps-sdk/jsx';\nimport {\n HandlerType,\n getJsxChildren,\n unwrapError,\n walkJsx,\n} from '@metamask/snaps-utils';\nimport { hasProperty } from '@metamask/utils';\nimport type { PayloadAction } from '@reduxjs/toolkit';\nimport { type SagaIterator } from 'redux-saga';\nimport { call, put, select, take } from 'redux-saga/effects';\n\nimport type { SnapInterface, SnapInterfaceActions } from '../../types';\nimport type { RootControllerMessenger } from './controllers';\nimport type { Interface, RunSagaFunction } from './store';\nimport { getCurrentInterface, resolveInterface, setInterface } from './store';\n\n/**\n * Get a user interface object from a type and content object.\n *\n * @param runSaga - A function to run a saga outside the usual Redux flow.\n * @param type - The type of the interface.\n * @param content - The content to show in the interface.\n * @param interfaceActions - The actions to interact with the interface.\n * @returns The user interface object.\n */\nexport function getInterfaceResponse(\n runSaga: RunSagaFunction,\n type: DialogType,\n content: JSXElement,\n interfaceActions: SnapInterfaceActions,\n): SnapInterface {\n switch (type) {\n case DialogType.Alert:\n return {\n ...interfaceActions,\n type,\n content,\n ok: resolveWith(runSaga, null),\n };\n\n case DialogType.Confirmation:\n return {\n ...interfaceActions,\n type,\n content,\n\n ok: resolveWith(runSaga, true),\n cancel: resolveWith(runSaga, false),\n };\n\n case DialogType.Prompt:\n return {\n ...interfaceActions,\n type,\n content,\n\n ok: resolveWithInput(runSaga),\n cancel: resolveWith(runSaga, null),\n };\n\n default:\n throw new Error(`Unknown or unsupported dialog type: \"${String(type)}\".`);\n }\n}\n\n/**\n * Resolve the current user interface with the given value. This returns a\n * function that can be used to resolve the user interface.\n *\n * @param runSaga - A function to run a saga outside the usual Redux flow.\n * @param value - The value to resolve the user interface with.\n * @returns A function that can be used to resolve the user interface.\n */\nfunction resolveWith(runSaga: RunSagaFunction, value: unknown) {\n /**\n * Resolve the current user interface with the given value.\n *\n * @yields Puts the resolve user interface action.\n */\n function* resolveWithSaga(): SagaIterator {\n yield put(resolveInterface(value));\n }\n\n return async () => {\n await runSaga(resolveWithSaga).toPromise();\n };\n}\n\n/**\n * Resolve the current user interface with the provided input. This returns a\n * function that can be used to resolve the user interface.\n *\n * @param runSaga - A function to run a saga outside the usual Redux flow.\n * @returns A function that can be used to resolve the user interface.\n */\nfunction resolveWithInput(runSaga: RunSagaFunction) {\n /**\n * Resolve the current user interface with the given value.\n *\n * @param value - The value to resolve the user interface with.\n * @yields Puts the resolve user interface action.\n */\n function* resolveWithSaga(value: string): SagaIterator {\n yield put(resolveInterface(value));\n }\n\n return async (value = '') => {\n await runSaga(resolveWithSaga, value).toPromise();\n };\n}\n\n/**\n * Get the stored user interface from the store.\n *\n * @param controllerMessenger - The controller messenger used to call actions.\n * @param snapId - The Snap ID.\n * @yields Takes the set interface action.\n * @returns The user interface object.\n */\nfunction* getStoredInterface(\n controllerMessenger: RootControllerMessenger,\n snapId: SnapId,\n): SagaIterator<Interface & { content: JSXElement }> {\n const currentInterface: Interface | null = yield select(getCurrentInterface);\n\n if (currentInterface) {\n const { content } = controllerMessenger.call(\n 'SnapInterfaceController:getInterface',\n snapId,\n currentInterface.id,\n );\n\n return { ...currentInterface, content };\n }\n\n const { payload }: PayloadAction<Interface> = yield take(setInterface.type);\n const { content } = controllerMessenger.call(\n 'SnapInterfaceController:getInterface',\n snapId,\n payload.id,\n );\n\n return { ...payload, content };\n}\n\n/**\n * A JSX element with a name.\n */\nexport type NamedJSXElement = JSXElement & { props: { name: string } };\n\n/**\n * Check if a JSX element is a JSX element with a given name.\n *\n * @param element - The JSX element.\n * @param name - The element name.\n * @returns True if the element is a JSX element with the given name, otherwise\n * false.\n */\nfunction isJSXElementWithName<Element extends JSXElement, Name extends string>(\n element: Element,\n name: Name,\n): element is Element & { props: { name: Name } } {\n return hasProperty(element.props, 'name') && element.props.name === name;\n}\n\n/**\n * Find an element inside a form element in a JSX tree.\n *\n * @param form - The form element.\n * @param name - The element name.\n * @returns An object containing the element and the form name if it's contained\n * in a form, otherwise undefined.\n */\nfunction getFormElement(form: FormElement, name: string) {\n const element = walkJsx<NamedJSXElement>(form, (childElement) => {\n if (isJSXElementWithName(childElement, name)) {\n return childElement;\n }\n\n return undefined;\n });\n\n if (element === undefined) {\n return undefined;\n }\n\n return { element, form: form.props.name };\n}\n\n/**\n * Get an element from a JSX tree with the given name.\n *\n * @param content - The interface content.\n * @param name - The element name.\n * @returns An object containing the element and the form name if it's contained\n * in a form, otherwise undefined.\n */\nexport function getElement(\n content: JSXElement,\n name: string,\n):\n | {\n element: NamedJSXElement;\n form?: string;\n }\n | undefined {\n if (isJSXElementWithName(content, name)) {\n return { element: content };\n }\n\n return walkJsx(content, (element) => {\n if (element.type === 'Form') {\n return getFormElement(element, name);\n }\n\n if (isJSXElementWithName(element, name)) {\n return { element };\n }\n\n return undefined;\n });\n}\n/**\n * Handle submitting event requests to OnUserInput including unwrapping potential errors.\n *\n * @param controllerMessenger - The controller messenger used to call actions.\n * @param snapId - The Snap ID.\n * @param id - The interface ID.\n * @param event - The event to submit.\n * @param context - The interface context.\n */\nasync function handleEvent(\n controllerMessenger: RootControllerMessenger,\n snapId: SnapId,\n id: string,\n event: UserInputEvent,\n context: InterfaceContext | null,\n) {\n try {\n await controllerMessenger.call(\n 'ExecutionService:handleRpcRequest',\n snapId,\n {\n origin: '',\n handler: HandlerType.OnUserInput,\n request: {\n jsonrpc: '2.0',\n method: ' ',\n params: {\n event,\n id,\n context,\n },\n },\n },\n );\n } catch (error) {\n const [unwrapped] = unwrapError(error);\n throw unwrapped;\n }\n}\n\n/**\n * Click on an element of the Snap interface.\n *\n * @param controllerMessenger - The controller messenger used to call actions.\n * @param id - The interface ID.\n * @param content - The interface content.\n * @param snapId - The Snap ID.\n * @param name - The element name.\n */\nexport async function clickElement(\n controllerMessenger: RootControllerMessenger,\n id: string,\n content: JSXElement,\n snapId: SnapId,\n name: string,\n): Promise<void> {\n const result = getElement(content, name);\n assert(\n result !== undefined,\n `Could not find an element in the interface with the name \"${name}\".`,\n );\n\n assert(\n result.element.type === 'Button',\n `Expected an element of type \"Button\", but found \"${result.element.type}\".`,\n );\n\n const { state, context } = controllerMessenger.call(\n 'SnapInterfaceController:getInterface',\n snapId,\n id,\n );\n\n // Button click events are always triggered.\n await handleEvent(\n controllerMessenger,\n snapId,\n id,\n {\n type: UserInputEventType.ButtonClickEvent,\n name: result.element.props.name,\n },\n context,\n );\n\n if (result.form && result.element.props.type === 'submit') {\n await handleEvent(\n controllerMessenger,\n snapId,\n id,\n {\n type: UserInputEventType.FormSubmitEvent,\n name: result.form,\n value: state[result.form] as Record<string, string | null>,\n },\n context,\n );\n }\n}\n\n/**\n * Merge a value in the interface state.\n *\n * @param state - The actual interface state.\n * @param name - The component name that changed value.\n * @param value - The new value.\n * @param form - The form name if the element is in one.\n * @returns The state with the merged value.\n */\nexport function mergeValue(\n state: InterfaceState,\n name: string,\n value: string | null,\n form?: string,\n): InterfaceState {\n if (form) {\n return {\n ...state,\n [form]: {\n ...(state[form] as FormState),\n [name]: value,\n },\n };\n }\n\n return { ...state, [name]: value };\n}\n\n/**\n * Type a value in an interface element.\n *\n * @param controllerMessenger - The controller messenger used to call actions.\n * @param id - The interface ID.\n * @param content - The interface Components.\n * @param snapId - The Snap ID.\n * @param name - The element name.\n * @param value - The value to type in the element.\n */\nexport async function typeInField(\n controllerMessenger: RootControllerMessenger,\n id: string,\n content: JSXElement,\n snapId: SnapId,\n name: string,\n value: string,\n) {\n const result = getElement(content, name);\n\n assert(\n result !== undefined,\n `Could not find an element in the interface with the name \"${name}\".`,\n );\n\n assert(\n result.element.type === 'Input',\n `Expected an element of type \"Input\", but found \"${result.element.type}\".`,\n );\n\n const { state, context } = controllerMessenger.call(\n 'SnapInterfaceController:getInterface',\n snapId,\n id,\n );\n\n const newState = mergeValue(state, name, value, result.form);\n\n controllerMessenger.call(\n 'SnapInterfaceController:updateInterfaceState',\n id,\n newState,\n );\n\n await controllerMessenger.call('ExecutionService:handleRpcRequest', snapId, {\n origin: '',\n handler: HandlerType.OnUserInput,\n request: {\n jsonrpc: '2.0',\n method: ' ',\n params: {\n event: {\n type: UserInputEventType.InputChangeEvent,\n name: result.element.props.name,\n value,\n },\n id,\n context,\n },\n },\n });\n}\n\n/**\n * Type a value in an interface element.\n *\n * @param controllerMessenger - The controller messenger used to call actions.\n * @param id - The interface ID.\n * @param content - The interface Components.\n * @param snapId - The Snap ID.\n * @param name - The element name.\n * @param value - The value to type in the element.\n */\nexport async function selectInDropdown(\n controllerMessenger: RootControllerMessenger,\n id: string,\n content: JSXElement,\n snapId: SnapId,\n name: string,\n value: string,\n) {\n const result = getElement(content, name);\n\n assert(\n result !== undefined,\n `Could not find an element in the interface with the name \"${name}\".`,\n );\n\n assert(\n result.element.type === 'Dropdown',\n `Expected an element of type \"Dropdown\", but found \"${result.element.type}\".`,\n );\n\n const options = getJsxChildren(result.element) as JSXElement[];\n const selectedOption = options.find(\n (option) =>\n hasProperty(option.props, 'value') && option.props.value === value,\n );\n\n assert(\n selectedOption !== undefined,\n `The dropdown with the name \"${name}\" does not contain \"${value}\".`,\n );\n\n const { state, context } = controllerMessenger.call(\n 'SnapInterfaceController:getInterface',\n snapId,\n id,\n );\n\n const newState = mergeValue(state, name, value, result.form);\n\n controllerMessenger.call(\n 'SnapInterfaceController:updateInterfaceState',\n id,\n newState,\n );\n\n await controllerMessenger.call('ExecutionService:handleRpcRequest', snapId, {\n origin: '',\n handler: HandlerType.OnUserInput,\n request: {\n jsonrpc: '2.0',\n method: ' ',\n params: {\n event: {\n type: UserInputEventType.InputChangeEvent,\n name: result.element.props.name,\n value,\n },\n id,\n context,\n },\n },\n });\n}\n\n/**\n * Get a user interface object from a Snap.\n *\n * @param runSaga - A function to run a saga outside the usual Redux flow.\n * @param snapId - The Snap ID.\n * @param controllerMessenger - The controller messenger used to call actions.\n * @yields Takes the set interface action.\n * @returns The user interface object.\n */\nexport function* getInterface(\n runSaga: RunSagaFunction,\n snapId: SnapId,\n controllerMessenger: RootControllerMessenger,\n): SagaIterator<SnapInterface> {\n const { type, id, content } = yield call(\n getStoredInterface,\n controllerMessenger,\n snapId,\n );\n\n const interfaceActions = {\n clickElement: async (name: string) => {\n await clickElement(controllerMessenger, id, content, snapId, name);\n },\n typeInField: async (name: string, value: string) => {\n await typeInField(controllerMessenger, id, content, snapId, name, value);\n },\n selectInDropdown: async (name: string, value: string) => {\n await selectInDropdown(\n controllerMessenger,\n id,\n content,\n snapId,\n name,\n value,\n );\n },\n };\n\n return getInterfaceResponse(runSaga, type, content, interfaceActions);\n}\n"]}
@@ -4,7 +4,7 @@ import {
4
4
  import {
5
5
  getCreateInterfaceImplementation,
6
6
  getGetInterfaceImplementation
7
- } from "./chunk-WH5C5WIZ.mjs";
7
+ } from "./chunk-FQWOVTBB.mjs";
8
8
  import {
9
9
  getShowInAppNotificationImplementation,
10
10
  getShowNativeNotificationImplementation
@@ -86,4 +86,4 @@ export {
86
86
  getPermissionSpecifications,
87
87
  getEndowments
88
88
  };
89
- //# sourceMappingURL=chunk-TAIJXTLU.mjs.map
89
+ //# sourceMappingURL=chunk-TMAWGVZP.mjs.map
@@ -3,7 +3,7 @@ import {
3
3
  } from "./chunk-XK5HPZZC.mjs";
4
4
  import {
5
5
  handleRequest
6
- } from "./chunk-QC3BL5R2.mjs";
6
+ } from "./chunk-UPR3PTSA.mjs";
7
7
  import {
8
8
  rootLogger
9
9
  } from "./chunk-J4ZPUCLX.mjs";
@@ -160,4 +160,4 @@ async function installSnap(snapId, options = {}) {
160
160
  export {
161
161
  installSnap
162
162
  };
163
- //# sourceMappingURL=chunk-R7PQRB54.mjs.map
163
+ //# sourceMappingURL=chunk-U3DVRTS2.mjs.map
@@ -1,17 +1,30 @@
1
1
  import {
2
2
  clickElement,
3
3
  getInterface,
4
+ selectInDropdown,
4
5
  typeInField
5
- } from "./chunk-B23SV5TL.mjs";
6
+ } from "./chunk-AFA4KKWW.mjs";
7
+ import {
8
+ SnapResponseStruct
9
+ } from "./chunk-HOI6FPLR.mjs";
6
10
  import {
7
11
  clearNotifications,
8
12
  getNotifications
9
13
  } from "./chunk-LB4R3BUA.mjs";
10
14
 
11
15
  // src/internals/request.ts
16
+ import {
17
+ ComponentOrElementStruct
18
+ } from "@metamask/snaps-sdk";
12
19
  import { unwrapError } from "@metamask/snaps-utils";
13
- import { getSafeJson, hasProperty, isPlainObject } from "@metamask/utils";
20
+ import {
21
+ assert,
22
+ getSafeJson,
23
+ hasProperty,
24
+ isPlainObject
25
+ } from "@metamask/utils";
14
26
  import { nanoid } from "@reduxjs/toolkit";
27
+ import { is } from "superstruct";
15
28
  function handleRequest({
16
29
  snapId,
17
30
  store,
@@ -21,6 +34,11 @@ function handleRequest({
21
34
  runSaga,
22
35
  request: { id = nanoid(), origin = "https://metamask.io", ...options }
23
36
  }) {
37
+ const getInterfaceError = () => {
38
+ throw new Error(
39
+ "Unable to get the interface from the Snap: The request to the Snap failed."
40
+ );
41
+ };
24
42
  const promise = executionService.handleRpcRequest(snapId, {
25
43
  origin,
26
44
  handler,
@@ -32,19 +50,31 @@ function handleRequest({
32
50
  }).then(async (result) => {
33
51
  const notifications = getNotifications(store.getState());
34
52
  store.dispatch(clearNotifications());
35
- const getInterfaceFn = await getInterfaceApi(
36
- result,
37
- snapId,
38
- controllerMessenger
39
- );
40
- return {
41
- id: String(id),
42
- response: {
43
- result: getSafeJson(result)
44
- },
45
- notifications,
46
- ...getInterfaceFn ? { getInterface: getInterfaceFn } : {}
47
- };
53
+ try {
54
+ const getInterfaceFn = await getInterfaceApi(
55
+ result,
56
+ snapId,
57
+ controllerMessenger
58
+ );
59
+ return {
60
+ id: String(id),
61
+ response: {
62
+ result: getSafeJson(result)
63
+ },
64
+ notifications,
65
+ ...getInterfaceFn ? { getInterface: getInterfaceFn } : {}
66
+ };
67
+ } catch (error) {
68
+ const [unwrappedError] = unwrapError(error);
69
+ return {
70
+ id: String(id),
71
+ response: {
72
+ error: unwrappedError.serialize()
73
+ },
74
+ notifications: [],
75
+ getInterface: getInterfaceError
76
+ };
77
+ }
48
78
  }).catch((error) => {
49
79
  const [unwrappedError] = unwrapError(error);
50
80
  return {
@@ -52,16 +82,24 @@ function handleRequest({
52
82
  response: {
53
83
  error: unwrappedError.serialize()
54
84
  },
55
- notifications: []
85
+ notifications: [],
86
+ getInterface: getInterfaceError
56
87
  };
57
88
  });
58
89
  promise.getInterface = async () => {
59
- return await runSaga(
90
+ const sagaPromise = runSaga(
60
91
  getInterface,
61
92
  runSaga,
62
93
  snapId,
63
94
  controllerMessenger
64
95
  ).toPromise();
96
+ const result = await Promise.race([promise, sagaPromise]);
97
+ if (is(result, SnapResponseStruct) && hasProperty(result.response, "error")) {
98
+ throw new Error(
99
+ `Unable to get the interface from the Snap: The returned interface may be invalid. The error message received was: ${result.response.error.message}`
100
+ );
101
+ }
102
+ return await sagaPromise;
65
103
  };
66
104
  return promise;
67
105
  }
@@ -70,6 +108,10 @@ async function getInterfaceFromResult(result, snapId, controllerMessenger) {
70
108
  return result.id;
71
109
  }
72
110
  if (isPlainObject(result) && hasProperty(result, "content")) {
111
+ assert(
112
+ is(result.content, ComponentOrElementStruct),
113
+ "The Snap returned an invalid interface."
114
+ );
73
115
  const id = await controllerMessenger.call(
74
116
  "SnapInterfaceController:createInterface",
75
117
  snapId,
@@ -112,6 +154,16 @@ async function getInterfaceApi(result, snapId, controllerMessenger) {
112
154
  name,
113
155
  value
114
156
  );
157
+ },
158
+ selectInDropdown: async (name, value) => {
159
+ await selectInDropdown(
160
+ controllerMessenger,
161
+ interfaceId,
162
+ content,
163
+ snapId,
164
+ name,
165
+ value
166
+ );
115
167
  }
116
168
  };
117
169
  };
@@ -124,4 +176,4 @@ export {
124
176
  getInterfaceFromResult,
125
177
  getInterfaceApi
126
178
  };
127
- //# sourceMappingURL=chunk-QC3BL5R2.mjs.map
179
+ //# sourceMappingURL=chunk-UPR3PTSA.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/internals/request.ts"],"sourcesContent":["import type { AbstractExecutionService } from '@metamask/snaps-controllers';\nimport {\n type SnapId,\n type JsonRpcError,\n type ComponentOrElement,\n ComponentOrElementStruct,\n} from '@metamask/snaps-sdk';\nimport type { HandlerType } from '@metamask/snaps-utils';\nimport { unwrapError } from '@metamask/snaps-utils';\nimport {\n assert,\n getSafeJson,\n hasProperty,\n isPlainObject,\n} from '@metamask/utils';\nimport { nanoid } from '@reduxjs/toolkit';\nimport { is } from 'superstruct';\n\nimport type {\n RequestOptions,\n SnapHandlerInterface,\n SnapRequest,\n} from '../types';\nimport {\n clearNotifications,\n clickElement,\n getInterface,\n getNotifications,\n typeInField,\n selectInDropdown,\n} from './simulation';\nimport type { RunSagaFunction, Store } from './simulation';\nimport type { RootControllerMessenger } from './simulation/controllers';\nimport { SnapResponseStruct } from './structs';\n\nexport type HandleRequestOptions = {\n snapId: SnapId;\n store: Store;\n executionService: AbstractExecutionService<unknown>;\n handler: HandlerType;\n controllerMessenger: RootControllerMessenger;\n runSaga: RunSagaFunction;\n request: RequestOptions;\n};\n\n/**\n * Send a JSON-RPC request to the Snap, and wrap the response in a\n * {@link SnapResponse} object.\n *\n * @param options - The request options.\n * @param options.snapId - The ID of the Snap to send the request to.\n * @param options.store - The Redux store.\n * @param options.executionService - The execution service to use to send the\n * request.\n * @param options.handler - The handler to use to send the request.\n * @param options.controllerMessenger - The controller messenger used to call actions.\n * @param options.runSaga - A function to run a saga outside the usual Redux\n * flow.\n * @param options.request - The request to send.\n * @param options.request.id - The ID of the request. If not provided, a random\n * ID will be generated.\n * @param options.request.origin - The origin of the request. Defaults to\n * `https://metamask.io`.\n * @returns The response, wrapped in a {@link SnapResponse} object.\n */\nexport function handleRequest({\n snapId,\n store,\n executionService,\n handler,\n controllerMessenger,\n runSaga,\n request: { id = nanoid(), origin = 'https://metamask.io', ...options },\n}: HandleRequestOptions): SnapRequest {\n const getInterfaceError = () => {\n throw new Error(\n 'Unable to get the interface from the Snap: The request to the Snap failed.',\n );\n };\n\n const promise = executionService\n .handleRpcRequest(snapId, {\n origin,\n handler,\n request: {\n jsonrpc: '2.0',\n id: 1,\n ...options,\n },\n })\n .then(async (result) => {\n const notifications = getNotifications(store.getState());\n store.dispatch(clearNotifications());\n\n try {\n const getInterfaceFn = await getInterfaceApi(\n result,\n snapId,\n controllerMessenger,\n );\n\n return {\n id: String(id),\n response: {\n result: getSafeJson(result),\n },\n notifications,\n ...(getInterfaceFn ? { getInterface: getInterfaceFn } : {}),\n };\n } catch (error) {\n const [unwrappedError] = unwrapError(error);\n return {\n id: String(id),\n response: {\n error: unwrappedError.serialize(),\n },\n notifications: [],\n getInterface: getInterfaceError,\n };\n }\n })\n .catch((error) => {\n const [unwrappedError] = unwrapError(error);\n\n return {\n id: String(id),\n response: {\n error: unwrappedError.serialize(),\n },\n notifications: [],\n getInterface: getInterfaceError,\n };\n }) as unknown as SnapRequest;\n\n promise.getInterface = async () => {\n const sagaPromise = runSaga(\n getInterface,\n runSaga,\n snapId,\n controllerMessenger,\n ).toPromise();\n const result = await Promise.race([promise, sagaPromise]);\n\n // If the request promise has resolved to an error, we should throw\n // instead of waiting for an interface that likely will never be displayed\n if (\n is(result, SnapResponseStruct) &&\n hasProperty(result.response, 'error')\n ) {\n throw new Error(\n `Unable to get the interface from the Snap: The returned interface may be invalid. The error message received was: ${\n (result.response.error as JsonRpcError).message\n }`,\n );\n }\n\n return await sagaPromise;\n };\n\n return promise;\n}\n\n/**\n * Get the interface ID from the result if it's available or create a new interface if the result contains static components.\n *\n * @param result - The handler result object.\n * @param snapId - The Snap ID.\n * @param controllerMessenger - The controller messenger.\n * @returns The interface ID or undefined if the result doesn't include content.\n */\nexport async function getInterfaceFromResult(\n result: unknown,\n snapId: SnapId,\n controllerMessenger: RootControllerMessenger,\n) {\n if (isPlainObject(result) && hasProperty(result, 'id')) {\n return result.id as string;\n }\n\n if (isPlainObject(result) && hasProperty(result, 'content')) {\n assert(\n is(result.content, ComponentOrElementStruct),\n 'The Snap returned an invalid interface.',\n );\n const id = await controllerMessenger.call(\n 'SnapInterfaceController:createInterface',\n snapId,\n result.content as ComponentOrElement,\n );\n\n return id;\n }\n\n return undefined;\n}\n\n/**\n * Get the response content from the SnapInterfaceController and include the interaction methods.\n *\n * @param result - The handler result object.\n * @param snapId - The Snap ID.\n * @param controllerMessenger - The controller messenger.\n * @returns The content components if any.\n */\nexport async function getInterfaceApi(\n result: unknown,\n snapId: SnapId,\n controllerMessenger: RootControllerMessenger,\n): Promise<(() => SnapHandlerInterface) | undefined> {\n const interfaceId = await getInterfaceFromResult(\n result,\n snapId,\n controllerMessenger,\n );\n\n if (interfaceId) {\n return () => {\n const { content } = controllerMessenger.call(\n 'SnapInterfaceController:getInterface',\n snapId,\n interfaceId,\n );\n\n return {\n content,\n clickElement: async (name) => {\n await clickElement(\n controllerMessenger,\n interfaceId,\n content,\n snapId,\n name,\n );\n },\n typeInField: async (name, value) => {\n await typeInField(\n controllerMessenger,\n interfaceId,\n content,\n snapId,\n name,\n value,\n );\n },\n selectInDropdown: async (name, value) => {\n await selectInDropdown(\n controllerMessenger,\n interfaceId,\n content,\n snapId,\n name,\n value,\n );\n },\n };\n };\n }\n\n return undefined;\n}\n"],"mappings":"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","names":[]}