storybook/lib/client-api/src/story_store.test.ts
Tom Coleman 533b1fc2e4
Change story update functions to be updateX
Thanks @shilman!

Co-Authored-By: Michael Shilman <shilman@users.noreply.github.com>
2020-03-04 17:23:04 +11:00

607 lines
20 KiB
TypeScript

// foo
import createChannel from '@storybook/channel-postmessage';
import { toId } from '@storybook/csf';
import addons, { mockChannel } from '@storybook/addons';
import Events from '@storybook/core-events';
import StoryStore, { ErrorLike } from './story_store';
import { defaultDecorateStory } from './decorators';
jest.mock('@storybook/node-logger', () => ({
logger: {
info: jest.fn(),
warn: jest.fn(),
error: jest.fn(),
},
}));
let channel;
beforeEach(() => {
channel = createChannel({ page: 'preview' });
});
// make a story and add it to the store
const addStoryToStore = (store, kind, name, storyFn, parameters = {}) =>
store.addStory(
{
kind,
name,
storyFn,
parameters,
id: toId(kind, name),
},
{
applyDecorators: defaultDecorateStory,
}
);
describe('preview.story_store', () => {
describe('extract', () => {
it('produces stories objects with inherited metadata', () => {
const store = new StoryStore({ channel });
store.addGlobalMetadata({ parameters: { global: 'global' }, decorators: [] });
store.addKindMetadata('a', { parameters: { kind: 'kind' }, decorators: [] });
addStoryToStore(store, 'a', '1', () => 0, { story: 'story' });
addStoryToStore(store, 'a', '2', () => 0);
addStoryToStore(store, 'b', '1', () => 0);
const extracted = store.extract();
// We need exact key ordering, even if in theory JS doesn't guarantee it
expect(Object.keys(extracted)).toEqual(['a--1', 'a--2', 'b--1']);
// content of item should be correct
expect(extracted['a--1']).toMatchObject({
id: 'a--1',
kind: 'a',
name: '1',
parameters: { global: 'global', kind: 'kind', story: 'story' },
});
});
});
describe('getRawStory', () => {
it('produces a story with inherited decorators applied', () => {
const store = new StoryStore({ channel });
const globalDecorator = jest.fn().mockImplementation(s => s());
store.addGlobalMetadata({ parameters: {}, decorators: [globalDecorator] });
const kindDecorator = jest.fn().mockImplementation(s => s());
store.addKindMetadata('a', { parameters: {}, decorators: [kindDecorator] });
const story = jest.fn();
addStoryToStore(store, 'a', '1', story);
const { getDecorated } = store.getRawStory('a', '1');
getDecorated()();
expect(globalDecorator).toHaveBeenCalled();
expect(kindDecorator).toHaveBeenCalled();
expect(story).toHaveBeenCalled();
});
});
describe('args', () => {
it('args is initialized to the value stored in parameters.args[name] || parameters.argType[name].defaultValue', () => {
const store = new StoryStore({ channel });
addStoryToStore(store, 'a', '1', () => 0, {
argTypes: {
arg1: { defaultValue: 'arg1' },
arg2: { defaultValue: 2 },
arg3: { defaultValue: { complex: { object: ['type'] } } },
arg4: {},
arg5: {},
},
args: {
arg2: 3,
arg4: 'foo',
arg6: false,
},
});
expect(store.getRawStory('a', '1').args).toEqual({
arg1: 'arg1',
arg2: 3,
arg3: { complex: { object: ['type'] } },
arg4: 'foo',
arg6: false,
});
});
it('updateStoryArgs changes the args of a story, per-key', () => {
const store = new StoryStore({ channel });
addStoryToStore(store, 'a', '1', () => 0);
expect(store.getRawStory('a', '1').args).toEqual({});
store.updateStoryArgs('a--1', { foo: 'bar' });
expect(store.getRawStory('a', '1').args).toEqual({ foo: 'bar' });
store.updateStoryArgs('a--1', { baz: 'bing' });
expect(store.getRawStory('a', '1').args).toEqual({ foo: 'bar', baz: 'bing' });
});
it('is passed to the story in the context', () => {
const storyFn = jest.fn();
const store = new StoryStore({ channel });
addStoryToStore(store, 'a', '1', storyFn);
store.updateStoryArgs('a--1', { foo: 'bar' });
store.getRawStory('a', '1').storyFn();
expect(storyFn).toHaveBeenCalledWith(
expect.objectContaining({
args: { foo: 'bar' },
})
);
});
it('updateStoryArgs emits STORY_ARGS_CHANGED', () => {
const onArgsChangedChannel = jest.fn();
const testChannel = mockChannel();
testChannel.on(Events.STORY_ARGS_CHANGED, onArgsChangedChannel);
const store = new StoryStore({ channel: testChannel });
addStoryToStore(store, 'a', '1', () => 0);
store.updateStoryArgs('a--1', { foo: 'bar' });
expect(onArgsChangedChannel).toHaveBeenCalledWith('a--1', { foo: 'bar' });
store.updateStoryArgs('a--1', { baz: 'bing' });
expect(onArgsChangedChannel).toHaveBeenCalledWith('a--1', { foo: 'bar', baz: 'bing' });
});
it('should update if the CHANGE_STORY_ARGS event is received', () => {
const testChannel = mockChannel();
const store = new StoryStore({ channel: testChannel });
addStoryToStore(store, 'a', '1', () => 0);
testChannel.emit(Events.CHANGE_STORY_ARGS, 'a--1', { foo: 'bar' });
expect(store.getRawStory('a', '1').args).toEqual({ foo: 'bar' });
});
it('passes args as the first argument to the story if `parameters.passArgsFirst` is true', () => {
const store = new StoryStore({ channel });
store.addKindMetadata('a', {
parameters: {
argTypes: {
a: { defaultValue: 1 },
},
},
decorators: [],
});
const storyOne = jest.fn();
addStoryToStore(store, 'a', '1', storyOne);
store.getRawStory('a', '1').storyFn();
expect(storyOne).toHaveBeenCalledWith(
expect.objectContaining({
args: { a: 1 },
parameters: expect.objectContaining({}),
})
);
const storyTwo = jest.fn();
addStoryToStore(store, 'a', '2', storyTwo, { passArgsFirst: true });
store.getRawStory('a', '2').storyFn();
expect(storyTwo).toHaveBeenCalledWith(
{ a: 1 },
expect.objectContaining({
args: { a: 1 },
parameters: expect.objectContaining({}),
})
);
});
});
describe('parameterEnhancer', () => {
it('allows you to alter parameters when stories are added', () => {
const store = new StoryStore({ channel });
const enhancer = jest.fn().mockReturnValue({ c: 'd' });
store.addParameterEnhancer(enhancer);
addStoryToStore(store, 'a', '1', () => 0, { a: 'b' });
expect(enhancer).toHaveBeenCalledWith(expect.objectContaining({ parameters: { a: 'b' } }));
expect(store.getRawStory('a', '1').parameters).toEqual({ a: 'b', c: 'd' });
});
it('recursively passes parameters to successive enhancers', () => {
const store = new StoryStore({ channel });
const firstEnhancer = jest.fn().mockReturnValue({ c: 'd' });
store.addParameterEnhancer(firstEnhancer);
const secondEnhancer = jest.fn().mockReturnValue({ e: 'f' });
store.addParameterEnhancer(secondEnhancer);
addStoryToStore(store, 'a', '1', () => 0, { a: 'b' });
expect(firstEnhancer).toHaveBeenCalledWith(
expect.objectContaining({ parameters: { a: 'b' } })
);
expect(secondEnhancer).toHaveBeenCalledWith(
expect.objectContaining({ parameters: { a: 'b', c: 'd' } })
);
expect(store.getRawStory('a', '1').parameters).toEqual({ a: 'b', c: 'd', e: 'f' });
});
it('does not merge subkeys of parameter enhancer results', () => {
const store = new StoryStore({ channel });
const firstEnhancer = jest.fn().mockReturnValue({ ns: { c: 'd' } });
store.addParameterEnhancer(firstEnhancer);
const secondEnhancer = jest.fn().mockReturnValue({ ns: { e: 'f' } });
store.addParameterEnhancer(secondEnhancer);
addStoryToStore(store, 'a', '1', () => 0, { ns: { a: 'b' } });
expect(firstEnhancer).toHaveBeenCalledWith(
expect.objectContaining({ parameters: { ns: { a: 'b' } } })
);
expect(secondEnhancer).toHaveBeenCalledWith(
expect.objectContaining({ parameters: { ns: { c: 'd' } } })
);
expect(store.getRawStory('a', '1').parameters).toEqual({ ns: { e: 'f' } });
});
it('allows you to alter parameters when stories are re-added', () => {
const store = new StoryStore({ channel });
addons.setChannel(channel);
const enhancer = jest.fn().mockReturnValue({ c: 'd' });
store.addParameterEnhancer(enhancer);
addStoryToStore(store, 'a', '1', () => 0, { a: 'b' });
enhancer.mockClear();
store.removeStoryKind('a');
addStoryToStore(store, 'a', '1', () => 0, { e: 'f' });
expect(enhancer).toHaveBeenCalledWith(expect.objectContaining({ parameters: { e: 'f' } }));
expect(store.getRawStory('a', '1').parameters).toEqual({ e: 'f', c: 'd' });
});
});
describe('storySort', () => {
it('sorts stories using given function', () => {
const parameters = {
options: {
// Test function does reverse alphabetical ordering.
storySort: (a: any, b: any): number =>
a[1].kind === b[1].kind
? 0
: -1 * a[1].id.localeCompare(b[1].id, undefined, { numeric: true }),
},
};
const store = new StoryStore({ channel });
addStoryToStore(store, 'a/a', '1', () => 0, parameters);
addStoryToStore(store, 'a/a', '2', () => 0, parameters);
addStoryToStore(store, 'a/b', '1', () => 0, parameters);
addStoryToStore(store, 'b/b1', '1', () => 0, parameters);
addStoryToStore(store, 'b/b10', '1', () => 0, parameters);
addStoryToStore(store, 'b/b9', '1', () => 0, parameters);
addStoryToStore(store, 'c', '1', () => 0, parameters);
const extracted = store.extract();
expect(Object.keys(extracted)).toEqual([
'c--1',
'b-b10--1',
'b-b9--1',
'b-b1--1',
'a-b--1',
'a-a--1',
'a-a--2',
]);
});
it('sorts stories alphabetically', () => {
const parameters = {
options: {
storySort: {
method: 'alphabetical',
},
},
};
const store = new StoryStore({ channel });
addStoryToStore(store, 'a/b', '1', () => 0, parameters);
addStoryToStore(store, 'a/a', '2', () => 0, parameters);
addStoryToStore(store, 'a/a', '1', () => 0, parameters);
addStoryToStore(store, 'c', '1', () => 0, parameters);
addStoryToStore(store, 'b/b10', '1', () => 0, parameters);
addStoryToStore(store, 'b/b9', '1', () => 0, parameters);
addStoryToStore(store, 'b/b1', '1', () => 0, parameters);
const extracted = store.extract();
expect(Object.keys(extracted)).toEqual([
'a-a--2',
'a-a--1',
'a-b--1',
'b-b1--1',
'b-b9--1',
'b-b10--1',
'c--1',
]);
});
it('sorts stories in specified order or alphabetically', () => {
const parameters = {
options: {
storySort: {
method: 'alphabetical',
order: ['b', ['bc', 'ba', 'bb'], 'a', 'c'],
},
},
};
const store = new StoryStore({ channel });
addStoryToStore(store, 'a/b', '1', () => 0, parameters);
addStoryToStore(store, 'a', '1', () => 0, parameters);
addStoryToStore(store, 'c', '1', () => 0, parameters);
addStoryToStore(store, 'b/bd', '1', () => 0, parameters);
addStoryToStore(store, 'b/bb', '1', () => 0, parameters);
addStoryToStore(store, 'b/ba', '1', () => 0, parameters);
addStoryToStore(store, 'b/bc', '1', () => 0, parameters);
addStoryToStore(store, 'b', '1', () => 0, parameters);
const extracted = store.extract();
expect(Object.keys(extracted)).toEqual([
'b--1',
'b-bc--1',
'b-ba--1',
'b-bb--1',
'b-bd--1',
'a--1',
'a-b--1',
'c--1',
]);
});
it('sorts stories in specified order or by configure order', () => {
const parameters = {
options: {
storySort: {
method: 'configure',
order: ['b', 'a', 'c'],
},
},
};
const store = new StoryStore({ channel });
addStoryToStore(store, 'a/b', '1', () => 0, parameters);
addStoryToStore(store, 'a', '1', () => 0, parameters);
addStoryToStore(store, 'c', '1', () => 0, parameters);
addStoryToStore(store, 'b/bd', '1', () => 0, parameters);
addStoryToStore(store, 'b/bb', '1', () => 0, parameters);
addStoryToStore(store, 'b/ba', '1', () => 0, parameters);
addStoryToStore(store, 'b/bc', '1', () => 0, parameters);
addStoryToStore(store, 'b', '1', () => 0, parameters);
const extracted = store.extract();
expect(Object.keys(extracted)).toEqual([
'b--1',
'b-bd--1',
'b-bb--1',
'b-ba--1',
'b-bc--1',
'a--1',
'a-b--1',
'c--1',
]);
});
});
describe('configuration', () => {
it('does not allow addStory if not configuring, unless allowUsafe=true', () => {
const store = new StoryStore({ channel });
store.finishConfiguring();
expect(() => addStoryToStore(store, 'a', '1', () => 0)).toThrow(
'Cannot add a story when not configuring'
);
expect(() =>
store.addStory(
{
kind: 'a',
name: '1',
storyFn: () => 0,
parameters: {},
id: 'a--1',
},
{
applyDecorators: defaultDecorateStory,
allowUnsafe: true,
}
)
).not.toThrow();
});
it('does not allow remove if not configuring, unless allowUsafe=true', () => {
const store = new StoryStore({ channel });
addons.setChannel(channel);
addStoryToStore(store, 'a', '1', () => 0);
store.finishConfiguring();
expect(() => store.remove('a--1')).toThrow('Cannot remove a story when not configuring');
expect(() => store.remove('a--1', { allowUnsafe: true })).not.toThrow();
});
it('does not allow removeStoryKind if not configuring, unless allowUsafe=true', () => {
const store = new StoryStore({ channel });
addons.setChannel(channel);
addStoryToStore(store, 'a', '1', () => 0);
store.finishConfiguring();
expect(() => store.removeStoryKind('a')).toThrow('Cannot remove a kind when not configuring');
expect(() => store.removeStoryKind('a', { allowUnsafe: true })).not.toThrow();
});
it('waits for configuration to be over before emitting SET_STORIES', () => {
const onSetStories = jest.fn();
channel.on(Events.SET_STORIES, onSetStories);
const store = new StoryStore({ channel });
addStoryToStore(store, 'a', '1', () => 0);
expect(onSetStories).not.toHaveBeenCalled();
store.finishConfiguring();
expect(onSetStories).toHaveBeenCalledWith({
stories: {
'a--1': expect.objectContaining({
id: 'a--1',
}),
},
});
});
it('emits an empty SET_STORIES if no stories were added during configuration', () => {
const onSetStories = jest.fn();
channel.on(Events.SET_STORIES, onSetStories);
const store = new StoryStore({ channel });
store.finishConfiguring();
expect(onSetStories).toHaveBeenCalledWith({ stories: {} });
});
it('allows configuration as second time (HMR)', () => {
const onSetStories = jest.fn();
channel.on(Events.SET_STORIES, onSetStories);
const store = new StoryStore({ channel });
store.finishConfiguring();
onSetStories.mockClear();
store.startConfiguring();
addStoryToStore(store, 'a', '1', () => 0);
store.finishConfiguring();
expect(onSetStories).toHaveBeenCalledWith({
stories: {
'a--1': expect.objectContaining({
id: 'a--1',
}),
},
});
});
});
describe('HMR behaviour', () => {
it('emits the right things after removing a story', () => {
const onSetStories = jest.fn();
channel.on(Events.SET_STORIES, onSetStories);
const store = new StoryStore({ channel });
// For hooks
addons.setChannel(channel);
store.startConfiguring();
addStoryToStore(store, 'kind-1', 'story-1.1', () => 0);
addStoryToStore(store, 'kind-1', 'story-1.2', () => 0);
store.finishConfiguring();
onSetStories.mockClear();
store.startConfiguring();
store.remove(toId('kind-1', 'story-1.1'));
store.finishConfiguring();
expect(onSetStories).toHaveBeenCalledWith({
stories: {
'kind-1--story-1-2': expect.objectContaining({
id: 'kind-1--story-1-2',
}),
},
});
expect(store.fromId(toId('kind-1', 'story-1.1'))).toBeFalsy();
expect(store.fromId(toId('kind-1', 'story-1.2'))).toBeTruthy();
});
it('emits the right things after removing a kind', () => {
const onSetStories = jest.fn();
channel.on(Events.SET_STORIES, onSetStories);
const store = new StoryStore({ channel });
// For hooks
addons.setChannel(channel);
store.startConfiguring();
addStoryToStore(store, 'kind-1', 'story-1.1', () => 0);
addStoryToStore(store, 'kind-1', 'story-1.2', () => 0);
addStoryToStore(store, 'kind-2', 'story-2.1', () => 0);
addStoryToStore(store, 'kind-2', 'story-2.2', () => 0);
store.finishConfiguring();
onSetStories.mockClear();
store.startConfiguring();
store.removeStoryKind('kind-1');
store.finishConfiguring();
expect(onSetStories).toHaveBeenCalledWith({
stories: {
'kind-2--story-2-1': expect.objectContaining({
id: 'kind-2--story-2-1',
}),
'kind-2--story-2-2': expect.objectContaining({
id: 'kind-2--story-2-2',
}),
},
});
expect(store.fromId(toId('kind-1', 'story-1.1'))).toBeFalsy();
expect(store.fromId(toId('kind-2', 'story-2.1'))).toBeTruthy();
});
// eslint-disable-next-line jest/expect-expect
it('should not error even if you remove a kind that does not exist', () => {
const store = new StoryStore({ channel });
store.removeStoryKind('kind');
});
});
describe('RENDER_CURRENT_STORY', () => {
it('is emitted when setError is called', () => {
const onRenderCurrentStory = jest.fn();
channel.on(Events.RENDER_CURRENT_STORY, onRenderCurrentStory);
const store = new StoryStore({ channel });
store.setError(new Error('Something is bad!') as ErrorLike);
expect(onRenderCurrentStory).toHaveBeenCalled();
});
it('is NOT emitted when setSelection is called during configuration', () => {
const onRenderCurrentStory = jest.fn();
channel.on(Events.RENDER_CURRENT_STORY, onRenderCurrentStory);
const store = new StoryStore({ channel });
store.setSelection({ storyId: 'a--1', viewMode: 'story' });
expect(onRenderCurrentStory).not.toHaveBeenCalled();
});
it('is emitted when configuration ends', () => {
const onRenderCurrentStory = jest.fn();
channel.on(Events.RENDER_CURRENT_STORY, onRenderCurrentStory);
const store = new StoryStore({ channel });
store.finishConfiguring();
expect(onRenderCurrentStory).toHaveBeenCalled();
});
it('is emitted when setSelection is called outside of configuration', () => {
const onRenderCurrentStory = jest.fn();
channel.on(Events.RENDER_CURRENT_STORY, onRenderCurrentStory);
const store = new StoryStore({ channel });
store.finishConfiguring();
onRenderCurrentStory.mockClear();
store.setSelection({ storyId: 'a--1', viewMode: 'story' });
expect(onRenderCurrentStory).toHaveBeenCalled();
});
});
});