5.1 KiB
Storybook Instrumenter
The Storybook Instrumenter is used to patch a (3rd party) module to track and intercept function invocations for step-through debugging using the Interactions addon. In essense, the Instrumenter traverses a given object, recursively monkey-patching any functions to make them "tracked".
During normal operation, tracked functions simply call through to their original function, forwarding the return value. As a side-effect, they also emit a call
event whenever they are invoked.
Through options
, functions can be marked "interceptable", which give them another mode of operation. In this "intercept" mode, the original function is not invoked, instead the interceptable function returns a Promise
which only resolves when receiving an event to do so. This enables step-through debugging, directly in the browser. A consequence of this design is that all interceptable functions must be await
-ed, even if their original function is not asynchronous (i.e. it normally does not return a Promise).
API
The primary way to use the Storybook Instrumenter is through the instrument
function:
instrument<TObj extends Record<string, any>>(obj: TObj, options: Options): TObj
instrument
takes a plain JS object or imported ES module, and optionally an options
object. It traverses the input object, recursively iterating over object properties and arrays. Any values with typeof function
are tracked (through monkey-patching). Finally, a shallow copy of the original object is returned (with functions replaced). If the mutate: true
option is set, the original object is mutated instead of returning a shallow copy.
Events
The Storybook Instrumenter uses the Storybook Channel API to send and receive events.
Emitted tracking events
The instrumenter emits two types of events for tracking function invocations ("calls"):
storybook/instrumenter/call
- Emitted whenever a tracked function is invokedstorybook/instrumenter/sync
- Emitted after one or more tracked functions are invoked (batch-wise)
The storybook/instrumenter/call
event payload contains all metadata about the function invocation, including a unique id
, any arguments, the method name and object path. However, the order of events is not guaranteed and you may receive the same call multiple times while debugging. Moreover, this event is emitted for all tracked calls, not just interceptable ones.
The storybook/instrumenter/sync
event payload contains a list of logItems
which represents a "normalized" log of interceptable calls. The order of calls is guaranteed and step-through debugging will not append to the log but rather update it to set the proper status
for each call. The log does not contain full call metadata but only a callId
, so this must be mapped onto received storybook/instrumenter/call
events. The storybook/instrumenter/sync
event also contains callStates
, see below.
Received control events
The instrumenter listens for these control events:
storybook/instrumenter/start
- Remount the story and start the debugger at the first interceptable callstorybook/instrumenter/back
- Remount the story and start the debugger at the previous interceptable callstorybook/instrumenter/goto
- Fast-forwards to - or remounts and starts debugging at - the given interceptable callstorybook/instrumenter/next
- Resolves the Promise for the currently intercepted call, letting execution continue to the next callstorybook/instrumenter/end
- Resolves all Promises for intercepted calls, letting execution continue to the end
Remounting is achieved through emitting Storybook's forceRemount
event. In some situations, this will trigger a full page refresh (of the preview) in order to flush pending promises (e.g. long-running interactions).
Control states
Besides patching functions, the instrumenter keeps track of "control states". These indicate whether the debugger is available, and which control events are available for use:
debugger: boolean
- Whether theinteractionsDebugger
feature flag is enabledstart: boolean
- Whether emittingstorybook/instrumenter/start
would workback: boolean
- Whether emittingstorybook/instrumenter/back
would workgoto: boolean
- Whether emittingstorybook/instrumenter/goto
would worknext: boolean
- Whether emittingstorybook/instrumenter/next
would workend: boolean
- Whether emittingstorybook/instrumenter/end
would work
These values are provided in the controlStates
object on the storybook/instrumenter/sync
event payload.
Options
intercept: boolean | ((method: string, path: Array<string | CallRef>) => boolean)
- Whether to make functions interceptableretain: boolean
- Whether to retain calls across renders (e.g. for story setup functions / loaders that run only once)mutate: boolean
- Whether to mutate the input object instead of returning a shallow copypath: Array<string | CallRef>
- A virtual object path to prepend to the actual input object function pathsgetArgs: (call: Call, state: State) => Call['args']
- Allows overriding args before invoking the original function with them