storybook/lib/store/src/StoryStore.ts
2021-09-25 14:16:05 +10:00

404 lines
13 KiB
TypeScript

import memoize from 'memoizerific';
import {
Parameters,
StoryId,
StoryContextForLoaders,
AnyFramework,
ProjectAnnotations,
ComponentTitle,
StoryContext,
} from '@storybook/csf';
import mapValues from 'lodash/mapValues';
import pick from 'lodash/pick';
import { StoryIndexStore } from './StoryIndexStore';
import { ArgsStore } from './ArgsStore';
import { GlobalsStore } from './GlobalsStore';
import { processCSFFile } from './processCSFFile';
import { prepareStory } from './prepareStory';
import {
CSFFile,
ModuleImportFn,
Story,
NormalizedProjectAnnotations,
Path,
ExtractOptions,
ModuleExports,
BoundStory,
} from './types';
import { HooksContext } from './hooks';
import { normalizeInputTypes } from './normalizeInputTypes';
import { inferArgTypes } from './inferArgTypes';
import { inferControls } from './inferControls';
type MaybePromise<T> = Promise<T> | T;
// TODO -- what are reasonable values for these?
const CSF_CACHE_SIZE = 1000;
const STORY_CACHE_SIZE = 10000;
function normalizeProjectAnnotations<TFramework extends AnyFramework>({
argTypes,
globalTypes,
argTypesEnhancers,
...annotations
}: ProjectAnnotations<TFramework>): NormalizedProjectAnnotations<TFramework> {
return {
...(argTypes && { argTypes: normalizeInputTypes(argTypes) }),
...(globalTypes && { globalTypes: normalizeInputTypes(globalTypes) }),
argTypesEnhancers: [
...(argTypesEnhancers || []),
inferArgTypes,
// inferControls technically should only run if the user is using the controls addon,
// and so should be added by a preset there. However, as it seems some code relies on controls
// annotations (in particular the angular implementation's `cleanArgsDecorator`), for backwards
// compatibility reasons, we will leave this in the store until 7.0
inferControls,
],
...annotations,
};
}
export class StoryStore<TFramework extends AnyFramework> {
storyIndex: StoryIndexStore;
importFn: ModuleImportFn;
projectAnnotations: NormalizedProjectAnnotations<TFramework>;
globals: GlobalsStore;
args: ArgsStore;
hooks: Record<StoryId, HooksContext<TFramework>>;
cachedCSFFiles?: Record<Path, CSFFile<TFramework>>;
processCSFFileWithCache: typeof processCSFFile;
prepareStoryWithCache: typeof prepareStory;
constructor({
importFn,
fetchStoryIndex,
}: {
importFn: ModuleImportFn;
fetchStoryIndex: ConstructorParameters<typeof StoryIndexStore>[0]['fetchStoryIndex'];
}) {
this.storyIndex = new StoryIndexStore({ fetchStoryIndex });
this.importFn = importFn;
this.globals = new GlobalsStore();
this.args = new ArgsStore();
this.hooks = {};
// We use a cache for these two functions for two reasons:
// 1. For performance
// 2. To ensure that when the same story is prepared with the same inputs you get the same output
this.processCSFFileWithCache = memoize(CSF_CACHE_SIZE)(processCSFFile) as typeof processCSFFile;
this.prepareStoryWithCache = memoize(STORY_CACHE_SIZE)(prepareStory) as typeof prepareStory;
}
// See note in PreviewWeb about the 'sync' init path.
initialize(options: {
projectAnnotations: ProjectAnnotations<TFramework>;
sync: false;
cacheAllCSFFiles?: boolean;
}): Promise<void>;
initialize(options: {
projectAnnotations: ProjectAnnotations<TFramework>;
sync: true;
cacheAllCSFFiles?: boolean;
}): void;
initialize({
projectAnnotations,
sync = false,
cacheAllCSFFiles = false,
}: {
projectAnnotations: ProjectAnnotations<TFramework>;
sync?: boolean;
cacheAllCSFFiles?: boolean;
}): MaybePromise<void> {
this.projectAnnotations = normalizeProjectAnnotations(projectAnnotations);
const { globals, globalTypes } = this.projectAnnotations;
this.globals.initialize({ globals, globalTypes });
if (sync) {
this.storyIndex.initialize({ sync: true });
if (cacheAllCSFFiles) {
this.cacheAllCSFFiles(true);
}
return null;
}
return this.storyIndex
.initialize({ sync: false })
.then(() => (cacheAllCSFFiles ? this.cacheAllCSFFiles(false) : null));
}
updateProjectAnnotations(projectAnnotations: ProjectAnnotations<TFramework>) {
this.projectAnnotations = normalizeProjectAnnotations(projectAnnotations);
const { globals, globalTypes } = projectAnnotations;
this.globals.resetOnProjectAnnotationsChange({ globals, globalTypes });
}
// This means that one of the CSF functions has changed.
async onImportFnChanged({ importFn }: { importFn: ModuleImportFn }) {
this.importFn = importFn;
// We need to refetch the stories list as it may have changed too
await this.storyIndex.cache(false);
if (this.cachedCSFFiles) {
await this.cacheAllCSFFiles(false);
}
}
// TODO -- call this when calling `onImportFnChanged` rather than calling directly
async onStoryIndexChanged() {
// We need to refetch the stories list as it may have changed too
await this.storyIndex.cache(false);
if (this.cachedCSFFiles) {
await this.cacheAllCSFFiles(false);
}
}
// To load a single CSF file to service a story we need to look up the importPath in the index
loadCSFFileByStoryId(storyId: StoryId, options: { sync: false }): Promise<CSFFile<TFramework>>;
loadCSFFileByStoryId(storyId: StoryId, options: { sync: true }): CSFFile<TFramework>;
loadCSFFileByStoryId(
storyId: StoryId,
{ sync = false }: { sync?: boolean } = {}
): MaybePromise<CSFFile<TFramework>> {
const { importPath, title } = this.storyIndex.storyIdToEntry(storyId);
const moduleExportsOrPromise = this.importFn(importPath);
const isPromise = Promise.resolve(moduleExportsOrPromise) === moduleExportsOrPromise;
if (!isPromise) {
// We pass the title in here as it may have been generated by autoTitle on the server.
return this.processCSFFileWithCache(moduleExportsOrPromise as ModuleExports, title);
}
if (sync) {
throw new Error(
`importFn() returned a promise, did you pass an async version then call initialize({sync: true})?`
);
}
return (moduleExportsOrPromise as Promise<ModuleExports>).then((moduleExports) =>
// We pass the title in here as it may have been generated by autoTitle on the server.
this.processCSFFileWithCache(moduleExports, title)
);
}
loadAllCSFFiles(sync: false): Promise<StoryStore<TFramework>['cachedCSFFiles']>;
loadAllCSFFiles(sync: true): StoryStore<TFramework>['cachedCSFFiles'];
loadAllCSFFiles(sync: boolean): MaybePromise<StoryStore<TFramework>['cachedCSFFiles']> {
const importPaths: Record<Path, StoryId> = {};
Object.entries(this.storyIndex.stories).forEach(([storyId, { importPath }]) => {
importPaths[importPath] = storyId;
});
const csfFileList = Object.entries(importPaths).map(([importPath, storyId]) => ({
importPath,
csfFileOrPromise: sync
? this.loadCSFFileByStoryId(storyId, { sync: true })
: this.loadCSFFileByStoryId(storyId, { sync: false }),
}));
function toObject(list: { importPath: Path; csfFile: CSFFile<TFramework> }[]) {
return list.reduce((acc, { importPath, csfFile }) => {
acc[importPath] = csfFile;
return acc;
}, {} as Record<Path, CSFFile<TFramework>>);
}
if (sync) {
return toObject(
csfFileList.map(({ importPath, csfFileOrPromise }) => ({
importPath,
csfFile: csfFileOrPromise,
})) as { importPath: Path; csfFile: CSFFile<TFramework> }[]
);
}
return Promise.all(
csfFileList.map(async ({ importPath, csfFileOrPromise }) => ({
importPath,
csfFile: await csfFileOrPromise,
}))
).then(toObject);
}
cacheAllCSFFiles(sync: false): Promise<void>;
cacheAllCSFFiles(sync: true): void;
cacheAllCSFFiles(sync: boolean): MaybePromise<void> {
if (sync) {
this.cachedCSFFiles = this.loadAllCSFFiles(true);
return null;
}
return this.loadAllCSFFiles(false).then((csfFiles) => {
this.cachedCSFFiles = csfFiles;
});
}
// Load the CSF file for a story and prepare the story from it and the project annotations.
async loadStory({ storyId }: { storyId: StoryId }): Promise<Story<TFramework>> {
const csfFile = await this.loadCSFFileByStoryId(storyId, { sync: false });
return this.storyFromCSFFile({ storyId, csfFile });
}
// This function is synchronous for convenience -- often times if you have a CSF file already
// it is easier not to have to await `loadStory`.
storyFromCSFFile({
storyId,
csfFile,
}: {
storyId: StoryId;
csfFile: CSFFile<TFramework>;
}): Story<TFramework> {
const storyAnnotations = csfFile.stories[storyId];
if (!storyAnnotations) {
throw new Error(`Didn't find '${storyId}' in CSF file, this is unexpected`);
}
const componentAnnotations = csfFile.meta;
const story = this.prepareStoryWithCache(
storyAnnotations,
componentAnnotations,
this.projectAnnotations
);
this.args.setInitial(story.id, story.initialArgs);
this.hooks[story.id] = this.hooks[story.id] || new HooksContext();
return story;
}
// If we have a CSF file we can get all the stories from it synchronously
componentStoriesFromCSFFile({ csfFile }: { csfFile: CSFFile<TFramework> }): Story<TFramework>[] {
return Object.keys(csfFile.stories).map((storyId: StoryId) =>
this.storyFromCSFFile({ storyId, csfFile })
);
}
// A prepared story does not include args, globals or hooks. These are stored in the story store
// and updated separtely to the (immutable) story.
getStoryContext(story: Story<TFramework>): Omit<StoryContextForLoaders<TFramework>, 'viewMode'> {
return {
...story,
args: this.args.get(story.id),
globals: this.globals.get(),
hooks: this.hooks[story.id] as unknown,
};
}
cleanupStory(story: Story<TFramework>): void {
this.hooks[story.id].clean();
}
extract(options: ExtractOptions = { includeDocsOnly: false }): Record<string, any> {
if (!this.cachedCSFFiles) {
throw new Error('Cannot call extract() unless you call cacheAllCSFFiles() first.');
}
return Object.entries(this.storyIndex.stories)
.map(([storyId, { importPath }]) => {
const csfFile = this.cachedCSFFiles[importPath];
const story = this.storyFromCSFFile({ storyId, csfFile });
if (!options.includeDocsOnly && story.parameters.docsOnly) {
return false;
}
return Object.entries(story).reduce(
(acc, [key, value]) => {
if (typeof value === 'function') {
return acc;
}
if (['hooks'].includes(key)) {
return acc;
}
if (Array.isArray(value)) {
return Object.assign(acc, { [key]: value.slice().sort() });
}
return Object.assign(acc, { [key]: value });
},
{ args: story.initialArgs }
);
})
.filter(Boolean);
}
getSetStoriesPayload() {
const stories = this.extract({ includeDocsOnly: true });
const kindParameters: Parameters = stories.reduce(
(acc: Parameters, { title }: { title: ComponentTitle }) => {
acc[title] = {};
return acc;
},
{} as Parameters
);
return {
v: 2,
globals: this.globals.get(),
globalParameters: {},
kindParameters,
stories,
};
}
getStoriesJsonData = () => {
const value = this.getSetStoriesPayload();
const allowedParameters = ['fileName', 'docsOnly', 'framework', '__id', '__isArgsStory'];
return {
v: 2,
globalParameters: pick(value.globalParameters, allowedParameters),
kindParameters: mapValues(value.kindParameters, (v) => pick(v, allowedParameters)),
stories: mapValues(value.stories, (v: any) => ({
...pick(v, ['id', 'name', 'kind', 'story']),
parameters: pick(v.parameters, allowedParameters),
})),
};
};
raw(): BoundStory<TFramework>[] {
return this.extract().map(({ id }: { id: StoryId }) => this.fromId(id));
}
fromId(storyId: StoryId): BoundStory<TFramework> {
if (!this.cachedCSFFiles) {
throw new Error('Cannot call fromId/raw() unless you call cacheAllCSFFiles() first.');
}
let importPath;
try {
({ importPath } = this.storyIndex.storyIdToEntry(storyId));
} catch (err) {
return null;
}
const csfFile = this.cachedCSFFiles[importPath];
const story = this.storyFromCSFFile({ storyId, csfFile });
return {
...story,
storyFn: (update) => {
const context = {
...this.getStoryContext(story),
viewMode: 'story',
} as StoryContext<TFramework>;
return story.unboundStoryFn({ ...context, ...update });
},
};
}
}