From 25f460d4a3b7ec36491b70ac47b5612bab4aa2a7 Mon Sep 17 00:00:00 2001 From: Elliot Winkler Date: Mon, 2 Sep 2024 04:32:47 -0600 Subject: [PATCH] Add guidelines for writing controllers (#4110) ## Explanation So far we have not have any documentation on how to write controllers the BaseController v2 way. These guidelines aim to fill in the missing pieces. Note that while each guideline has examples, we are still missing a complete example controller file which demonstrates all of the guidelines in one go. That will come in a future PR. ## References Progresses #4504. ## Changelog (None) ## Checklist - [ ] I've updated the test suite for new or updated code as appropriate - [ ] I've updated documentation (JSDoc, Markdown, etc.) for new or updated code as appropriate - [ ] I've highlighted breaking changes using the "BREAKING" category above as appropriate --------- Co-authored-by: Jongsun Suh Co-authored-by: Michele Esposito <34438276+mikesposito@users.noreply.github.com> --- docs/writing-controllers.md | 1489 +++++++++++++++++++++++++++++++++++ 1 file changed, 1489 insertions(+) create mode 100644 docs/writing-controllers.md diff --git a/docs/writing-controllers.md b/docs/writing-controllers.md new file mode 100644 index 0000000000..d8df3cb2e5 --- /dev/null +++ b/docs/writing-controllers.md @@ -0,0 +1,1489 @@ +# Guidelines for Writing Controllers + +## Understand the purpose of the controller pattern + +Controllers are foundational pieces within MetaMask's architecture: + +- They keep and manage wallet-centric data, which can be saved and recalled upon reloading (accounts, transactions, preferences, etc.). +- They contain business logic that powers functionality in the product (what happens when the wallet is locked, the network is switched, a transaction is completed, etc.). +- They act as a communication layer between service layers (blockchains, internal or external APIs, etc.). +- They allow the application to be divided into logical modules which can be maintained by different teams within the company. + +## Use the latest version of `BaseController` for controllers + +All controllers should inherit from `BaseController` from the `@metamask/base-controller` package. This provides a few benefits: + +- It defines a standard interface for all controllers. +- It introduces the messenger system, which is useful for interacting with other controllers without requiring direct access to them. +- It enforces that `update` is the only way to modify the state of the controller and provides a way to listen for state updates via the messenger. +- It simplifies initialization by consolidating constructor arguments into one options object. + +## Don't use `BaseController` for non-controllers + +One of the uniquely identifying features of a controller is the ability to manage state. + +If you have a class that does not capture any data in state, then your class does not need to inherit from `BaseController` (even if it uses the messaging system). + +## Maintain a clear and concise API + +The name of the controller should reflect its responsibility. If, when creating a controller, it is difficult to come up with a good name, consider defining the responsibility first. + +Each public method and each state property of a controller should have a purpose, and the name of the method or state property should be readable and should reflect the purpose clearly. If something does not need to be public, it should be made private; if it is unnecessary, it should be removed. + +## Accept an optional, partial representation of state + +Although `BaseController` requires a full representation of controller state, in practice, controllers should accept a partial version and then supply missing properties with defaults. In fact, the `state` argument should be optional: + +```typescript +type FooControllerState = { + // ... +}; + +function getDefaultFooControllerState(): FooControllerState { + // ... +} + +class FooController extends BaseController { + constructor({ + messenger, + state = {}, + }: { + messenger: FooControllerMessenger; + state: Partial; + }) { + super({ + // ... + messenger, + state: { ...getDefaultFooControllerState(), ...state }, + }); + + // ... + } +} +``` + +## Provide a default representation of state + +Each controller needs a default representation in order to fully initialize itself when [receiving a partial representation of state](#accept-a-partial-representation-of-state). A default representation of state is also useful when testing interactions with a controller's `*:stateChange` event. + +A function which returns this default representation should be defined and exported. It should be called `getDefault${ControllerName}State`. + +Note that the default representation object is not exported directly to ensure that it is a new object reference each time it is requested, thereby preventing accidental updates to this object in tests or other places. + +```typescript +/* === packages/foo-controller/src/FooController.ts === */ + +type FooControllerState = { + // ... +}; + +export function getDefaultFooControllerState(): FooControllerState { + // ... +} + +class FooController extends BaseController { + constructor({ + messenger, + state = {}, + }: { + messenger: FooControllerMessenger; + state: Partial; + }) { + super({ + name, + messenger, + state: { ...getDefaultFooControllerState(), ...state }, + }); + + // ... + } +} + +/* === packages/foo-controller/src/index.ts === */ + +export { FooController, getDefaultFooControllerState } from './FooController'; +``` + +## Define metadata for state properties + +Each property in state has two pieces of metadata that must be specified. This instructs the client how to treat that property: + +- `persist` — Informs the client whether the property should be placed in persistent storage (`true`) or not (`false`). Opting out is useful if you want to have a property in state for convenience reasons but you know that property is ephemeral and can be easily reconstructed. +- `anonymous` — Informs the client whether the property is free of personally identifiable information (`true`) or not (`false`) and can therefore safely be included and sent to error reporting services such as Sentry. When in doubt, use `false`. + +A variable named `${controllerName}Metadata` should be defined (there is no need to export it) and passed as the `metadata` argument in the constructor to `BaseController`. + +```typescript +const keyringControllerMetadata = { + vault: { + // We want to persist this property so it's restored automatically, as we + // cannot reconstruct it otherwise. + persist: true, + // This property can be used to identify a user, so we want to make sure we + // do not include it in Sentry. + anonymous: false, + }, + isUnlocked: { + // We do not need to persist this property in state, as we want to + // initialize state with the wallet unlocked. + persist: false, + // This property has no PII, so it is safe to send to Sentry. + anonymous: true, + }, +}; + +class KeyringController extends BaseController { + constructor(/* ... */) { + super({ + // name: ..., + metadata: keyringControllerMetadata, + // messenger: ..., + // state: ..., + }); + } +} +``` + +## Use single "options bag" for constructor arguments + +A controller should receive all of its arguments as named argument via a single "options bag". These arguments must include those that are required by `BaseController` (`messenger`, `metadata`, `name`, and `state`). However, they may also include any that are required by the controller itself; there is no need for additional positional arguments after the options bag: + +🚫 **`isEnabled` is a separate argument** + +```typescript +class FooController extends BaseController { + constructor( + { + messenger, + state = {}, + }: { + messenger: FooControllerMessenger; + state: Partial; + }, + isEnabled: boolean, + ) { + // ... + } +} +``` + +✅ **`isEnabled` is another option** + +```typescript +class FooController extends BaseController { + constructor({ + messenger, + state = {}, + isEnabled, + }: { + messenger: FooControllerMessenger; + state: Partial; + isEnabled: boolean; + }) { + // ... + } +} +``` + +## Use the messaging system instead of callbacks + +Prior to BaseController v2, it was common for a controller to respond to an event occurring within another controller (such a state change) by receiving an event listener callback which the client would bind ahead of time: + +🚫 **The constructor takes a callback function, `onBarStateChange`** + +```typescript +/* === This repo: packages/foo-controller/src/FooController.ts === */ + +class FooController extends BaseControllerV1 { + constructor({ + onBarStateChange, + }, { + onBarStateChange: (state: BarControllerState) => void, + }) { + onBarStateChange((state) => { + // do something with the state + }); + } +} + +// === Client repo === + +const barController = new BarController(); +const fooController = new FooController({ + onBarStateChange: barController.subscribe.bind(barController), +}); +``` + +If the recipient controller supports the messaging system, however, the callback pattern is unnecessary. Using the messenger not only aligns the controller with `BaseController`, but also reduces the number of options that consumers need to remember in order to use the controller: + +✅ **The constructor subscribes to the `BarController:stateChange` event** + +```typescript +/* === This repo: packages/foo-controller/src/FooController.ts === */ + +const name = 'FooController'; + +type FooControllerMessenger = RestrictedControllerMessenger< + typeof name, + never, + never, + never, + never +>; + +class FooController extends BaseController< + 'FooController', + // ..., + FooControllerMessenger +> { + constructor({ messenger /*, ... */ }, { messenger: FooControllerMessenger }) { + super({ messenger /* ... */ }); + + messenger.subscribe('BarController:stateChange', (state) => { + // do something with the state + }); + } +} + +// === Client repo === + +const rootMessenger = new ControllerMessenger< + 'BarController:stateChange', + never +>(); +const barControllerMessenger = rootMessenger.getRestricted({ + name: 'BarController', +}); +const barController = new BarController({ + messenger: barControllerMessenger, +}); +const fooControllerMessenger = rootMessenger.getRestricted({ + name: 'FooController', +}); +const fooController = new FooController({ + messenger: fooControllerMessenger, +}); +``` + +## Use the messaging system instead of event emitters + +Some controllers expose an EventEmitter object so that other parts of the system can listen to them: + +🚫 **The controller emits the `someEvent` event** + +```typescript +/* === This repo: packages/foo-controller/src/FooController.ts === */ + +import { EventEmitter } from 'events'; + +class FooController extends BaseController { + hub: EventEmitter; + + constructor(/* ... */) { + // ... + + this.hub = new EventEmitter(); + } + + doSomething() { + this.hub.emit('someEvent'); + } +} + +// === Client repo === + +const fooController = new FooController(); +fooController.hub.on('someEvent', () => { + // respond to the event somehow +}); +``` + +However, this pattern can be replaced with the use of the messenger: + +✅ **The controller emits the `FooController:someEvent` event** + +```typescript +/* === This repo: packages/foo-controller/src/FooController.ts === */ + +const name = 'FooController'; + +type FooControllerMessenger = RestrictedControllerMessenger< + typeof name, + never, + never, + never, + never +>; + +class FooController extends BaseController< + 'FooController', + // ..., + FooControllerMessenger +> { + constructor({ messenger /*, ... */ }, { messenger: FooControllerMessenger }) { + super({ messenger /*, ... */ }); + } + + doSomething() { + this.messagingSystem.publish('FooController:someEvent'); + } +} + +// === Client repo === + +const rootMessenger = new ControllerMessenger< + 'FooController:someEvent', + never +>(); +const fooControllerMessenger = rootMessenger.getRestricted({ + name: 'FooController', +}); +const fooController = new FooController({ + messenger: fooControllerMessenger, +}); +rootMessenger.subscribe('FooController:someEvent', () => { + // do something with the event +}); +``` + +## Define the `*:getState` action using the `ControllerGetStateAction` utility type + +Each controller needs a type for its `*:getState` action. The `ControllerGetStateAction` utility type from the `@metamask/base-controller` package should be used to define this type. + +The name of this type should be `${ControllerName}GetStateAction`. + +```typescript +import type { ControllerGetStateAction } from '@metamask/base-controller'; + +type FooControllerGetStateAction = ControllerGetStateAction< + 'FooController', + FooControllerState +>; +``` + +## Define the `*:stateChange` event using the `ControllerStateChangeEvent` utility type + +Each controller needs a type for its `*:stateChange` event. The `ControllerStateChangeEvent` utility type from the `@metamask/base-controller` package should be used to define this type. + +The name of this type should be `${ControllerName}StateChangeEvent`. + +```typescript +import type { ControllerStateChangeEvent } from '@metamask/base-controller'; + +type FooControllerStateChangeEvent = ControllerStateChangeEvent< + 'FooController', + FooControllerState +>; +``` + +## Use standard naming scheme for action identifiers + +An action is equivalent to a function, and its identifier should be named in "camelCase". + +```typescript +const name = 'FooController'; + +export type FooControllerShowNotificationAction = { + type: `${typeof name}:showNotification`; + handler: () => void; +}; +``` + +## Use standard naming scheme for action types + +Types for messenger actions should follow this naming scheme: + +``` +Controller name + Action identifier without controller name prefix + "Action" +``` + +🚫 **Type name does not include controller name and does not end with "Action"** + +```typescript +const name = 'FooController'; + +export type ShowNotification = { + type: `${typeof name}:show`; + handler: () => void; +}; +``` + +🚫 **Type name includes controller name, but does not match action identifier minus prefix** + +```typescript +const name = 'FooController'; + +export type FooControllerShowNotificationAction = { + type: `${typeof name}:show`; + handler: () => void; +}; +``` + +✅ **Type name now matches controller name + action identifier + "Action"** + +```typescript +const name = 'FooController'; + +export type FooControllerShowAction = { + type: `${typeof name}:show`; + handler: () => void; +}; +``` + +## Use standard naming scheme for event identifiers + +An event is not an action, so the identifier for an event should not describe something that the controller needs to do, but rather something that has has happened to the controller. + +You may find it helpful to pretend that the word "on" precedes the identifier (but do not include this in the identifier itself). + +🚫 **`receiveMessage` sounds like a command** + +```typescript +const name = 'FooController'; + +export type FooControllerReceiveMessageEvent = { + type: `${typeof name}:receiveMessage`; + payload: () => void; +}; +``` + +🚫 **The event identifier includes "on"** + +```typescript +const name = 'FooController'; + +export type FooControllerOnMessageReceivedEvent = { + type: `${typeof name}:onMessageReceived`; + payload: () => void; +}; +``` + +✅ **The event identifier sounds like a statement and does not include "on"** + +```typescript +const name = 'FooController'; + +export type FooControllerMessageReceivedEvent = { + type: `${typeof name}:messageReceived`; + payload: () => void; +}; +``` + +## Use standard naming scheme for event types + +Types for messenger events should follow this naming: + +``` +Controller name + Event identifier without controller name prefix + "Event" +``` + +🚫 **Type name does not include controller name and does not end with "Action"** + +```typescript +const name = 'FooController'; + +export type MessageReceived = { + type: `${typeof name}:messageReceived`; + payload: () => void; +}; +``` + +✅ **Type name now matches controller name + event identifier + "Event"** + +```typescript +const name = 'FooController'; + +export type FooControllerMessageReceivedEvent = { + type: `${typeof name}:messageReceived`; + payload: () => void; +}; +``` + +## Define and export a type union for internal action types + +A controller should define and export a type union that holds all of its actions. This type union makes it easy to define the type for the controller's messenger. + +The name of this type should be `${ControllerName}Actions`. + +This type should be only passed to `RestrictedControllerMessenger` as the 2nd type parameter. It should _not_ be included in its 4th type parameter, as that is is used for external events. + +🚫 **`FooController['type']` is passed as the 4th type parameter** + +```typescript +export type FooControllerActions = + | FooControllerUpdateCurrencyAction + | FooControllerUpdateRatesAction; + +export type FooControllerMessenger = RestrictedControllerMessenger< + 'FooController', + FooControllerActions, + never, + FooControllerActions['type'], + never +>; +``` + +✅ **`never` is passed as the 5th type parameter (assuming no external events)** + +```typescript +export type FooControllerActions = + | FooControllerUpdateCurrencyAction + | FooControllerUpdateRatesAction; + +export type FooControllerMessenger = RestrictedControllerMessenger< + 'FooController', + FooControllerActions, + never, + never, + never +>; +``` + +## Define and export a type union for internal event types + +A controller should define and export a type union that holds all of its events. This type union makes it easy to define the type for the controller's messenger. + +The name of this type should be `${ControllerName}Events`. + +This type should be only passed to `RestrictedControllerMessenger` as the 3rd type parameter. It should _not_ be included in its 5th type parameter, as that is is used for external events. + +🚫 **`FooControllerEvents['type']` is passed as the 5th type parameter** + +```typescript +export type FooControllerEvents = + | FooControllerMessageReceivedEvent + | FooControllerNotificationAddedEvent; + +export type FooControllerMessenger = RestrictedControllerMessenger< + 'FooController', + never, + FooControllerEvents, + never, + FooControllerEvents['type'] +>; +``` + +✅ **`never` is passed as the 5th type parameter (assuming no external events)** + +```typescript +export type FooControllerEvents = + | FooControllerMessageReceivedEvent + | FooControllerNotificationAddedEvent; + +export type FooControllerMessenger = RestrictedControllerMessenger< + 'FooController', + never, + FooControllerEvents, + never, + never +>; +``` + +## Define, but do not export, a type union for external action types + +A controller may wish to call actions defined by other controllers, and therefore will need to define them in the messenger's allowlist. + +In this case, the controller should group these types into a type union so that they can be easily passed to the `RestrictedControllerMessenger` type. However, it should not export this type, as it would then be re-exporting types that another package has already exported. + +The name of this type should be `AllowedActions`. + +This type should not only be passed to `RestrictedControllerMessenger` as the 2nd type parameter, but should also be included in its 4th type parameter. + +🚫 **`never` is passed as the 4th type parameter** + +```typescript +export type AllowedActions = + | BarControllerDoSomethingAction + | BarControllerDoSomethingElseAction; + +export type FooControllerMessenger = RestrictedControllerMessenger< + 'FooController', + AllowedActions, + never, + never, + never +>; +``` + +🚫 **`AllowedActions['type']` is passed as the 4th type parameter, but `AllowedActions` is exported** + +```typescript +/* === packages/foo-controller/src/FooController.ts === */ + +export type AllowedActions = + | BarControllerDoSomethingAction + | BarControllerDoSomethingElseAction; + +export type FooControllerMessenger = RestrictedControllerMessenger< + 'FooController', + AllowedActions, + never, + AllowedActions['type'], + never +>; + +/* === packages/foo-controller/src/index.ts === */ + +export type { AllowedActions } from '@metamask/foo-controller'; +``` + +✅ **`AllowedActions['type']` is passed as the 4th type parameter, and `AllowedActions` is _not_ exported** + +```typescript +type AllowedActions = + | BarControllerDoSomethingAction + | BarControllerDoSomethingElseAction; + +export type FooControllerMessenger = RestrictedControllerMessenger< + 'FooController', + AllowedActions, + never, + AllowedActions['type'], + never +>; +``` + +If, in a test, you need to access all of the actions included in a controller's messenger allowlist, use the [`ExtractAvailableAction` utility type](../packages/base-controller/tests/helpers.ts): + +```typescript +// NOTE: You may need to adjust the path depending on where you are +import { ExtractAvailableAction } from '../../base-controller/tests/helpers'; + +const messenger = new ControllerMessenger< + ExtractAvailableAction, + never +>(); +``` + +## Define, but do not export, a type union for external event types + +A controller may wish to subscribe to events defined by other controllers, and therefore will need to define them in the messenger's allowlist. + +In this case, the controller should group these types into a type union so that they can be easily passed to the `RestrictedControllerMessenger` type. However, it should not export this type, as it would then be re-exporting types that another package has already exported. + +The name of this type should be `AllowedEvents`. + +This type should not only be passed to `RestrictedControllerMessenger` as the 3rd type parameter, but should also be included in its 5th type parameter. + +🚫 **`never` is passed as the 5th type parameter** + +```typescript +export type AllowedEvents = + | BarControllerSomethingHappenedEvent + | BarControllerSomethingElseHappenedEvent; + +export type FooControllerMessenger = RestrictedControllerMessenger< + 'FooController', + never, + AllowedEvents, + never, + never +>; +``` + +🚫 **`AllowedEvents['type']` is passed as the 5th type parameter, but `AllowedEvents` is exported** + +```typescript +/* === packages/foo-controller/src/FooController.ts === */ + +export type AllowedEvents = + | BarControllerSomethingHappenedEvent + | BarControllerSomethingElseHappenedEvent; + +export type FooControllerMessenger = RestrictedControllerMessenger< + 'FooController', + never, + AllowedEvents, + never, + AllowedEvents['type'] +>; + +/* === packages/foo-controller/src/index.ts === */ + +export type { AllowedEvents } from '@metamask/foo-controller'; +``` + +✅ **`AllowedEvents['type']` is passed as the 5th type parameter, and `AllowedEvents` is _not_ exported** + +```typescript +type AllowedEvents = + | BarControllerSomethingHappenedEvent + | BarControllerSomethingElseHappenedEvent; + +export type FooControllerMessenger = RestrictedControllerMessenger< + 'FooController', + never, + AllowedEvents, + never, + AllowedEvents['type'] +>; +``` + +If, in a test, you need to access all of the events included in a controller's messenger allowlist, use the [`ExtractAvailableEvent` utility type](../packages/base-controller/tests/helpers.ts): + +```typescript +// NOTE: You may need to adjust the path depending on where you are +import { ExtractAvailableEvent } from '../../base-controller/tests/helpers'; + +const messenger = new ControllerMessenger< + never, + ExtractAvailableEvent +>(); +``` + +## Define and export a type for the controller's messenger + +This type should include: + +- Actions defined and exported by the controller that it registers and expects consumers to call (i.e., _internal actions_) + - This should always include `${controllerName}GetStateAction` +- Actions imported from other controllers that the controller calls (i.e., _external actions_) +- Events defined and exported by the controller that it publishes and expects consumers to subscribe to (i.e., _internal events_) + - This should always include `${controllerName}StateChangeEvent` +- Events imported from other controllers that the controller subscribes to (i.e., _external events_) + +The name of this type should be `${ControllerName}Messenger`. + +A messenger with internal and external actions and events looks like this: + +```typescript +import type { + ApprovalControllerAddApprovalRequestAction, + ApprovalControllerAcceptApprovalRequestAction, + ApprovalControllerApprovalRequestApprovedEvent, + ApprovalControllerApprovalRequestRejectedEvent, +} from '@metamask/approval-controller'; + +export type SwapsControllerGetStateAction = ControllerGetStateAction< + 'SwapsController', + SwapsControllerState +>; + +export type SwapsControllerInitiateSwapAction = { + type: 'SwapsController:initiateSwap'; + handler: (swap: Swap) => void; +}; + +export type SwapsControllerActions = + | SwapsControllerGetStateAction + | SwapsControllerInitiateSwapAction; + +export type AllowedActions = + | ApprovalControllerAddApprovalRequestAction + | ApprovalControllerAcceptApprovalRequestAction; + +export type SwapsControllerStateChangeEvent = ControllerStateChangeEvent< + 'SwapsController', + SwapsControllerState +>; + +export type SwapsControllerSwapCreatedEvent = { + type: 'SwapsController:swapCreated'; + payload: (swap: Swap) => void; +}; + +export type SwapsControllerEvents = + | SwapsControllerStateChangeEvent + | SwapsControllerSwapCreatedEvent; + +export type AllowedEvents = + | ApprovalControllerApprovalRequestApprovedEvent + | ApprovalControllerApprovalRequestRejectedEvent; + +export type SwapsControllerMessenger = RestrictedControllerMessenger< + 'SwapsController', + SwapsControllerActions | AllowedActions, + SwapsControllerEvents | AllowedEvents, + AllowedActions['type'], + AllowedEvents['type'] +>; +``` + +A messenger that allows no actions or events (whether internal or external) looks like this: + +```typescript +export type SwapsControllerMessenger = RestrictedControllerMessenger< + 'SwapsController', + never, + never, + never, + never +>; +``` + +## Define and export a type for the controller's state + +The name of this type should be `${ControllerName}State`. It should be exported. + +🚫 **Non-standard name** + +```typescript +export type FooState = { + bar: string; +}; +``` + +🚫 **Not exported** + +```typescript +type FooControllerState = { + bar: string; +}; +``` + +✅ **Standard name and exported** + +```typescript +// === packages/foo-controller/src/FooController.ts + +export type FooControllerState = { + bar: string; +}; + +// === packages/foo-controller/src/index.ts + +export type { FooControllerState } from './FooController'; +``` + +The type should be compatible with `Json` (otherwise, a type error will occur when passing the state type as a type parameter to `BaseController`). For instance, if a property needs to be designed as optional, only the `?` token should be used to do so; `| undefined` cannot be used, as `undefined` is not part of the `Json` type. + +🚫 **`FooControllerState` uses `undefined` to denote an optional property** + +```typescript +export type FooControllerState = { + bar: string | undefined; +}; + +export class FooController extends BaseController< + 'FooController', + // ERROR: + // [tsserver 2344] [E] Type 'NftControllerState' does not satisfy the constraint 'StateConstraint'. + // Property 'bar' is incompatible with index signature. + // Type 'string | undefined' is not assignable to type 'Json'. + FooControllerState, + FooControllerMessenger +> { + // ... +} +``` + +✅ **`FooControllerState` uses `?` to denote an optional property** + +```typescript +export type FooControllerState = { + bar?: string; +}; + +export class FooController extends BaseController< + 'FooController', + // No error + FooControllerState, + FooControllerMessenger +> { + // ... +} +``` + +## Use selectors to reduce the scope of state change listeners + +Sometimes a controller needs to do something when a certain property in another controller's state changes. It is common to simply listen to the `stateChange` event of that controller and make use of whatever properties are necessary to perform the requisite action. However, this naive approach can lead to an unnecessarily frequent number of operations such as API calls: + +🚫 **`this.#updateGasFees` is called when any property in `NetworkController` state changes, even though only `selectedNetworkClientId` is used** + +```typescript +class GasFeeController extends BaseController { + constructor({ + messenger, + }: // ... + { + messenger: GasFeeControllerMessenger; + // ... + }) { + // ... + + messenger.subscribe( + 'NetworkController:stateChange', + (networkControllerState) => { + this.#updateGasFees(networkControllerState.selectedNetworkClientId); + }, + ); + } +} +``` + +One way to fix this is to check if the other controller (the one being subscribed to) has a more suitable, granular event for the data being acted upon. For instance, `NetworkController` has a `networkDidChange` event which can be used in place of `NetworkController:stateChange` if the subscribing controller needs to know when the network has been switched: + +✅ **`NetworkController:networkDidChange` is used instead of `NetworkController:stateChange`** + +```typescript +class GasFeeController extends BaseController { + constructor({ + messenger, + }: // ... + { + messenger: GasFeeControllerMessenger; + // ... + }) { + // ... + + messenger.subscribe( + 'NetworkController:networkDidChange', + (networkControllerState) => { + this.#updateGasFees(networkControllerState.selectedNetworkClientId); + }, + ); + } +} +``` + +If the target controller does not have an alternate event to use, another solution is to keep track of the old value of the property in question and only take action when it is different. This requires first getting the respective property from the target controller and then comparing it against the new property in the event handler: + +⚠️ + +```typescript +class TokensController extends BaseController { + constructor({ + messenger, + }: // ... + { + messenger: TokensControllerMessenger; + // ... + }) { + // ... + + const accountsController = messenger.call('AccountsController:getState'); + let selectedAccount = accountsController.internalAccounts.selectedAccount; + + messenger.subscribe( + 'AccountsController:stateChange', + (newAccountsControllerState) => { + if (newAccountsControllerState.selectedAccount !== selectedAccount) { + this.#updateTokens( + newAccountsControllerState.internalAccounts.selectedAccount, + ); + } + }, + ); + } +} +``` + +But this gets unwieldy if there are multiple properties to check: + +⚠️ + +```typescript +class NftController extends BaseController/*<...>*/ { + constructor({ messenger }, { messenger: NftControllerMessenger }) { + // ... + + let preferencesControllerState = messenger.call( + 'PreferencesController:getState', + ); + + messenger.subscribe( + 'PreferencesController:stateChange', + (newPreferencesControllerState) => { + if ( + preferencesControllerState.ipfsGateway !== newPreferencesControllerState.ipfsGateway, + preferencesControllerState.openSeaEnabled !== newPreferencesControllerState.openSeaEnabled, + preferencesControllerState.isIpfsGatewayEnabled !== newPreferencesControllerState.isIpfsGatewayEnabled, + ) { + this.#updateNfts( + newPreferencesControllerState.ipfsGateway, + newPreferencesControllerState.openSeaEnabled, + newPreferencesControllerState.isIpfsGatewayEnabled + ); + preferencesControllerState = newPreferencesControllerState; + } + }, + ); + } +} +``` + +Instead, you can define a selector function which returns a subset of the state you need to perform the action and pass it as the third argument to `subscribe`. Note that the return value of this function should not create a new object if none of the properties you are watching have changed. To aid with this, you can have your target controller [expose selectors of its own](#expose-derived-state-using-selectors-instead-of-getters) for each of the desired properties, and then you can use the `createSelector` function from the `reselect` library to compose those selectors. This ensures that your event handler is called at the right time: + +✅ + +```typescript +/* === packages/preferences-controller/src/PreferencesController.ts === */ + +const selectIpfsGateway = (preferencesControllerState) => + preferencesControllerState.ipfsGateway; + +const selectOpenSeaEnabled = (preferencesControllerState) => + preferencesControllerState.openSeaEnabled; + +const selectIsIpfsGatewayEnabled = (preferencesControllerState) => + preferencesControllerState.isIpfsGatewayEnabled; + +export const preferencesControllerSelectors = { + selectIpfsGateway, + selectOpenSeaEnabled, + selectIsIpfsGatewayEnabled, +}; + +/* === packages/nft-controller/src/NftController.ts === */ + +import { createSelector } from 'reselect'; +import { preferencesControllerSelectors } from '@metamask/preferences-controller'; + +const selectPreferencesControllerState = createSelector( + [ + preferencesControllerSelectors.selectIpfsGateway, + preferencesControllerSelectors.selectOpenSeaEnabled, + preferencesControllerSelectors.selectIsIpfsGatewayEnabled, + ], + (ipfsGateway, openSeaEnabled, isIpfsGatewayEnabled) => ({ + ipfsGateway, + openSeaEnabled, + isIpfsGatewayEnabled, + }), +); + +class NftController extends BaseController /*<...>*/ { + constructor({ messenger }, { messenger: NftControllerMessenger }) { + // ... + + messenger.subscribe( + 'PreferencesController:stateChange', + ({ ipfsGateway, openSeaEnabled, isIpfsGatewayEnabled }) => { + this.#updateNfts(ipfsGateway, openSeaEnabled, isIpfsGatewayEnabled); + }, + selectPreferencesControllerState, + ); + } +} +``` + +## Don't represent data in state multiple ways + +Ideally, a piece of data should be represented in only one way and kept in only one place in state. Storing copies or variations of the same data makes it confusing for consumers and makes it possible for different parts of the client to use state in an inconsistent manner, which could lead to strange behavior at best and fund loss issues at worst. Use types and validations to ensure that state is always coherent at both compile time and runtime. Remove higher level forms from state in favor of deriving them using other means such as [selectors](#expose-derived-state-using-selectors-instead-of-getters) or a self-subscription. + +🚫 **Message count is stored in state two ways** + +```typescript +type MessagesControllerState = { + messages: Message[]; + messageCount: number; +}; + +class MessagesController extends BaseController< + 'MessagesController', + MessagesControllerState + // ... +> { + addMessage(message: Message) { + this.update((state) => { + // Oops, we forgot to update message count! + state.messages.push(message); + }); + } +} +``` + +✅ **Only one way to access message count** + +```typescript +type MessagesControllerState = { + messages: Message[]; +}; + +class MessagesController extends BaseController< + 'MessagesController', + MessagesControllerState + // ... +> { + addMessage(message: Message) { + this.update((state) => { + // No need to also access message count, since it may be accessed via + // `state.messages.count` + state.messages.push(message); + }); + } +} +``` + +Similarly, for the same reasons, only one controller should "own" a piece of data: + +🚫 **Selected account is stored in two different controllers** + +```typescript +/* === packages/accounts-controller/src/AccountsController.ts === */ + +type Account = { + address: Hex; +} + +type AccountsControllerState = { + selectedAccount: Account; +} + +class AccountsController extends BaseController< + 'AccountsController', + AccountsControllerState, + // ... +> { + // ... +} + +/* === packages/preferences-controller/src/PreferencesController.ts === */ + +type PreferencesControllerState = { + selectedAccountAddress: Hex; +} + +class PreferencesController extends BaseController< + 'PreferencesController', + PreferencesControllerState, + ... +> { + // ... +} +``` + +✅ **Only one controller owns the selected account, and the other consumes it** + +```typescript +/* === packages/accounts-controller/src/AccountsController.ts === */ + +type Account = { + address: Hex; +}; + +type AccountsControllerState = { + selectedAccount: Account; +}; + +export type AccountsControllerGetStateAction = ControllerGetStateAction< + 'AccountsController', + AccountsControllerState +>; + +class AccountsController extends BaseController< + 'AccountsController', + AccountsControllerState + // ... +> { + // ... +} + +/* === packages/preferences-controller/src/PreferencesController.ts === */ + +import { AccountsControllerGetStateAction } from '@metamask/accounts-controller'; + +type AllowedActions = AccountsControllerGetStateAction; + +type PreferencesControllerMessenger = RestrictedControllerMessenger< + 'PreferencesController', + AllowedActions, + never, + AllowedActions['type'], + never +>; + +class PreferencesController extends BaseController< + 'PreferencesController', + // ... + PreferencesControllerMessenger +> { + constructor({ + messenger, + }: // ... + { + messenger: PreferencesControllerMessenger; + // ... + }) { + // ... + + messenger.subscribe( + 'AccountsController:getState', + (selectedAccount) => { + // do something with the selected account + }, + (accountsControllerState) => accountsControllerState.selectedAccount, + ); + } +} +``` + +### Also see + +- ["Keep state minimal and derive additional values"](https://redux.js.org/style-guide/#keep-state-minimal-and-derive-additional-values) in the Redux Style Guide. + +## Expose derived state using selectors instead of getters + +Sometimes, for convenience, consumers want access to a higher-level representation of a controller's state. It is tempting to add a method to the controller which provides this representation, but this means that a consumer would need an entire instance of the controller on hand to use this method. Using the messaging system mitigates this problem, but then the consumer would need access to a messenger as well, which may be impossible in places like Redux selector functions. + +To make it easier to share such representations across disparate parts of the codebase in a flexible fashion, you can define and export selector functions from your controller file instead. They should be placed under a `${controllerName}Selectors` object and then exported. + +🚫 **Convenience methods added to controller to access parts of state** + +```typescript +/* === packages/accounts-controller/src/AccountsController.ts === */ + +class AccountsController extends BaseController { + getActiveAccounts() { + return this.state.accounts.filter((account) => account.isActive); + } + + getInactiveAccounts() { + return this.state.accounts.filter((account) => account.isActive); + } +} + +/* === packages/tokens-controller/src/TokensController.ts === */ + +class TokensController extends BaseController { + constructor({ + accountsController, + }: // ... + { + accountsController: AccountsController; + // ... + }) { + this.#accountsController = accountsController; + // ... + } + + fetchTokens() { + const tokens = getTokens(this.#accountsController.getActiveAccounts()); + // ... do something with tokens ... + } +} +``` + +🚫 **Methods exposed via the messaging system** + +```typescript +/* === This repo: packages/accounts-controller/src/AccountsController.ts === */ + +export type AccountsControllerGetActiveAccountsAction = { + type: 'AccountsController:getActiveAccounts'; + handler: AccountsController['getActiveAccounts']; +}; + +export type AccountsControllerGetInactiveAccountsAction = { + type: 'AccountsController:getInactiveAccounts'; + handler: AccountsController['getInactiveAccounts']; +}; + +type AccountsControllerActions = + | AccountsControllerGetActiveAccountAction + | AccountsControllerGetInactiveAccountsAction; + +export type AccountsControllerMessenger = RestrictedControllerMessenger< + 'AccountsController', + AccountsControllerActions, + never, + never, + never +>; + +class AccountsController extends BaseController { + getActiveAccounts() { + return this.state.accounts.filter((account) => account.isActive); + } + + getInactiveAccounts() { + return this.state.accounts.filter((account) => account.isActive); + } +} + +/* === This repo: packages/tokens-controller/src/TokensController.ts === */ + +import type { + AccountsControllerGetActiveAccountsAction, + AccountsControllerGetInactiveAccountsAction, +} from '@metamask/accounts-controller'; + +type AllowedActions = + | AccountsControllerGetActiveAccountsAction + | AccountsControllerGetInactiveAccountsAction; + +export type TokensControllerMessenger = RestrictedControllerMessenger< + 'TokensController', + AllowedActions, + never, + AllowedActions['type'], + never +>; + +class TokensController extends BaseController { + constructor({ + messenger, + }: // ... + { + messenger: TokensControllerMessenger; + // ... + }) { + // ... + } + + fetchTokens() { + // Now TokensController no longer needs an instance of AccountsController to + // access the list of active accounts, which is good... + const tokens = getTokens( + this.messagingSystem.call('AccountsController:getActiveAccounts'), + ); + // ... do something with tokens ... + } +} + +/* === Client repo: selectors.ts === */ + +import { createSelector } from 'reselect'; +import { accountsControllerSelectors } from '@metamask/accounts-controller'; + +const selectAccountsControllerState = (state) => { + return state.engine.backgroundState.AccountsController; +}; + +// ⚠️ ... but this won't work! +export const selectActiveAccounts = createSelector( + selectAccountsControllerState, + (accountsControllerState) => accountsControllerState.getActiveAccounts(), +); +``` + +✅ **`reselect` used to select parts of state; selectors placed under object** + +```typescript +/* === This repo: packages/accounts-controller/src/AccountsController.ts === */ + +import { createSelector } from 'reselect'; + +type AccountsControllerState = { + accounts: Account[]; +}; + +const selectAccounts = (state: AccountsControllerState) => state.accounts; + +const selectActiveAccounts: createSelector( + [selectAccounts], + (accounts) => accounts.filter((account) => account.isActive), +); + +const selectInactiveAccounts: createSelector( + [selectAccounts], + (accounts) => accounts.filter((account) => !account.isActive), +); + +export const accountsControllerSelectors = { + selectAccounts, + selectActiveAccounts, + selectInactiveAccounts, +}; + +export type AccountsControllerGetStateAction = ControllerGetStateAction< + 'AccountsController', + AccountsControllerState, +>; + +type AccountsControllerActions = AccountsControllerGetStateAccountAction; + +export type AccountsControllerMessenger = RestrictedControllerMessenger< + 'AccountsController', + AccountsControllerActions, + never, + never, + never +>; + +/* === This repo: packages/tokens-controller/src/TokensController.ts === */ + +import type { + AccountsControllerGetStateAction, +} from '@metamask/accounts-controller'; +import { accountsControllerSelectors } from '@metamask/accounts-controller'; + +type AllowedActions = AccountsControllerGetStateAction; + +export type TokensControllerMessenger = RestrictedControllerMessenger< + 'TokensController', + AllowedActions, + never, + AllowedActions['type'], + never +>; + +class TokensController extends BaseController { + constructor({ + messenger, + // ... + }: { + messenger: TokensControllerMessenger, + // ... + }) { + // ... + } + + fetchTokens() { + // Now TokensController can use the selector in combination with the state + const tokensControllerState = this.messagingSystem.call( + 'AccountsController:getState', + ); + const accounts = accountsControllerSelectors.selectActiveAccounts( + tokensControllerState, + ); + const tokens = getTokens(accounts); + // ... do something with tokens ... + } +} + +/* === Client repo: selectors.ts === */ + +import { createSelector } from 'reselect'; +import { accountsControllerSelectors } from '@metamask/accounts-controller'; + +const selectAccountsControllerState = (state) => { + return state.engine.backgroundState.AccountsController; +}; + +// ⚠️ ... and this works too! +export const selectActiveAccounts = createSelector( + selectAccountsControllerState, + accountsControllerSelectors.selectActiveAccounts, +); +``` + +## Treat state-mutating methods as actions + +Just as each property of state [does not require a getter method to be accessed](#remove-getters-in-favor-of-direct-state-access), each property of state does not require a setter method to be updated, either. + +Methods that change the state of the controller do not need to represent granular, low-level operations such as adding, updating, or deleting a single property from state. Rather, they should be designed to support a higher-level task that the consumer wants to carry out. This is ultimately dictated by the needs of the client UI, and so they should also be given a name that reflects the behavior in the UI. + +🚫 **`setAlertShown` is too low-level** + +```typescript +class AlertsController extends BaseController { + setAlertShown(alertId: string, isShown: boolean) { + this.update((state) => { + state[alertId].isShown = isShown; + }); + } +} +``` + +✅ **An action is added for each distinct user interaction** + +```typescript +class AlertsController extends BaseController { + showAlert(alertId: string) { + this.update((state) => { + state.alerts[alertId].isShown = true; + }); + } + + hideAlert(alertId: string) { + this.update((state) => { + state.alerts[alertId].isShown = false; + }); + } +} +``` + +### Also see + +- ["Model actions as events, not setters"](https://redux.js.org/style-guide/#model-actions-as-events-not-setters) in the Redux Style Guide.