mirror of
https://github.com/storybookjs/storybook.git
synced 2025-04-06 15:31:16 +08:00
Add an extract
function to PreviewWeb
This commit is contained in:
parent
6ef1aef94b
commit
db60041745
@ -4,8 +4,8 @@ import merge from 'lodash/merge';
|
||||
import Events, { IGNORED_EXCEPTION } from '@storybook/core-events';
|
||||
import { logger } from '@storybook/client-logger';
|
||||
import addons, { mockChannel as createMockChannel } from '@storybook/addons';
|
||||
import { ModuleImportFn } from '@storybook/store';
|
||||
import { AnyFramework } from '@storybook/csf';
|
||||
import type { ModuleImportFn } from '@storybook/store';
|
||||
|
||||
import { PreviewWeb } from './PreviewWeb';
|
||||
import {
|
||||
@ -2725,4 +2725,159 @@ describe('PreviewWeb', () => {
|
||||
);
|
||||
});
|
||||
});
|
||||
|
||||
describe('extract', () => {
|
||||
// NOTE: if you are using storyStoreV6, and your `preview.js` throws, we do not currently
|
||||
// detect it (as we do not wrap the import of `preview.js` in a `try/catch`). The net effect
|
||||
// of that is that the `PreviewWeb`/`StoryStore` end up in an uninitalized state.
|
||||
it('throws an error if the preview is uninitialized', async () => {
|
||||
const preview = new PreviewWeb();
|
||||
await expect(preview.extract()).rejects.toThrow(/Failed to initialize/);
|
||||
});
|
||||
|
||||
it('throws an error if preview.js throws', async () => {
|
||||
const err = new Error('meta error');
|
||||
const preview = new PreviewWeb();
|
||||
await expect(
|
||||
preview.initialize({
|
||||
importFn,
|
||||
getProjectAnnotations: () => {
|
||||
throw err;
|
||||
},
|
||||
})
|
||||
).rejects.toThrow(err);
|
||||
|
||||
await expect(preview.extract()).rejects.toThrow(err);
|
||||
});
|
||||
|
||||
it('shows an error if the stories.json endpoint 500s', async () => {
|
||||
const err = new Error('sort error');
|
||||
mockFetchResult = { status: 500, text: async () => err.toString() };
|
||||
|
||||
const preview = new PreviewWeb();
|
||||
await expect(preview.initialize({ importFn, getProjectAnnotations })).rejects.toThrow(
|
||||
'sort error'
|
||||
);
|
||||
|
||||
await expect(preview.extract()).rejects.toThrow('sort error');
|
||||
});
|
||||
|
||||
it('waits for stories to be cached', async () => {
|
||||
const [gate, openGate] = createGate();
|
||||
|
||||
const gatedImportFn = async (path) => {
|
||||
await gate;
|
||||
return importFn(path);
|
||||
};
|
||||
|
||||
const preview = await createAndRenderPreview({ importFn: gatedImportFn });
|
||||
|
||||
let extracted = false;
|
||||
preview.extract().then(() => {
|
||||
extracted = true;
|
||||
});
|
||||
|
||||
expect(extracted).toBe(false);
|
||||
|
||||
openGate();
|
||||
await new Promise((r) => setTimeout(r, 0)); // Let the promise resolve
|
||||
expect(extracted).toBe(true);
|
||||
|
||||
expect(await preview.extract()).toMatchInlineSnapshot(`
|
||||
Object {
|
||||
"component-one--a": Object {
|
||||
"argTypes": Object {
|
||||
"foo": Object {
|
||||
"name": "foo",
|
||||
"type": Object {
|
||||
"name": "string",
|
||||
},
|
||||
},
|
||||
},
|
||||
"args": Object {
|
||||
"foo": "a",
|
||||
},
|
||||
"component": undefined,
|
||||
"componentId": "component-one",
|
||||
"id": "component-one--a",
|
||||
"initialArgs": Object {
|
||||
"foo": "a",
|
||||
},
|
||||
"kind": "Component One",
|
||||
"name": "A",
|
||||
"parameters": Object {
|
||||
"__isArgsStory": false,
|
||||
"docs": Object {
|
||||
"container": [MockFunction],
|
||||
},
|
||||
"fileName": "./src/ComponentOne.stories.js",
|
||||
},
|
||||
"story": "A",
|
||||
"subcomponents": undefined,
|
||||
"title": "Component One",
|
||||
},
|
||||
"component-one--b": Object {
|
||||
"argTypes": Object {
|
||||
"foo": Object {
|
||||
"name": "foo",
|
||||
"type": Object {
|
||||
"name": "string",
|
||||
},
|
||||
},
|
||||
},
|
||||
"args": Object {
|
||||
"foo": "b",
|
||||
},
|
||||
"component": undefined,
|
||||
"componentId": "component-one",
|
||||
"id": "component-one--b",
|
||||
"initialArgs": Object {
|
||||
"foo": "b",
|
||||
},
|
||||
"kind": "Component One",
|
||||
"name": "B",
|
||||
"parameters": Object {
|
||||
"__isArgsStory": false,
|
||||
"docs": Object {
|
||||
"container": [MockFunction],
|
||||
},
|
||||
"fileName": "./src/ComponentOne.stories.js",
|
||||
},
|
||||
"story": "B",
|
||||
"subcomponents": undefined,
|
||||
"title": "Component One",
|
||||
},
|
||||
"component-two--c": Object {
|
||||
"argTypes": Object {
|
||||
"foo": Object {
|
||||
"name": "foo",
|
||||
"type": Object {
|
||||
"name": "string",
|
||||
},
|
||||
},
|
||||
},
|
||||
"args": Object {
|
||||
"foo": "c",
|
||||
},
|
||||
"component": undefined,
|
||||
"componentId": "component-two",
|
||||
"id": "component-two--c",
|
||||
"initialArgs": Object {
|
||||
"foo": "c",
|
||||
},
|
||||
"kind": "Component Two",
|
||||
"name": "C",
|
||||
"parameters": Object {
|
||||
"__isArgsStory": false,
|
||||
"fileName": "./src/ComponentTwo.stories.js",
|
||||
},
|
||||
"playFunction": undefined,
|
||||
"story": "C",
|
||||
"subcomponents": undefined,
|
||||
"title": "Component Two",
|
||||
},
|
||||
}
|
||||
`);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
@ -80,6 +80,8 @@ export class PreviewWeb<TFramework extends AnyFramework> {
|
||||
|
||||
renderToDOM: WebProjectAnnotations<TFramework>['renderToDOM'];
|
||||
|
||||
previewEntryError?: Error;
|
||||
|
||||
previousSelection: Selection;
|
||||
|
||||
previousStory: Story<TFramework>;
|
||||
@ -294,6 +296,8 @@ export class PreviewWeb<TFramework extends AnyFramework> {
|
||||
}: {
|
||||
getProjectAnnotations: () => MaybePromise<ProjectAnnotations<TFramework>>;
|
||||
}) {
|
||||
delete this.previewEntryError;
|
||||
|
||||
const projectAnnotations = await this.getProjectAnnotationsOrRenderError(getProjectAnnotations);
|
||||
if (!this.storyStore.projectAnnotations) {
|
||||
await this.initializeWithProjectAnnotations(projectAnnotations);
|
||||
@ -306,6 +310,8 @@ export class PreviewWeb<TFramework extends AnyFramework> {
|
||||
}
|
||||
|
||||
async onStoryIndexChanged() {
|
||||
delete this.previewEntryError;
|
||||
|
||||
if (!this.storyStore.projectAnnotations) {
|
||||
// We haven't successfully set project annotations yet,
|
||||
// we need to do that before we can do anything else.
|
||||
@ -709,6 +715,28 @@ export class PreviewWeb<TFramework extends AnyFramework> {
|
||||
};
|
||||
}
|
||||
|
||||
// API
|
||||
async extract(options?: { includeDocsOnly: boolean }) {
|
||||
if (this.previewEntryError) {
|
||||
throw this.previewEntryError;
|
||||
}
|
||||
|
||||
if (!this.storyStore.projectAnnotations) {
|
||||
// In v6 mode, if your preview.js throws, we never get a chance to initialize the preview
|
||||
// or store, and the error is simply logged to the browser console. This is the best we can do
|
||||
throw new Error(dedent`Failed to initialize Storybook.
|
||||
|
||||
Do you have an error in your \`preview.js\`? Check your Storybook's browser console for errors.`);
|
||||
}
|
||||
|
||||
if (global.FEATURES?.storyStoreV7) {
|
||||
await this.storyStore.cacheAllCSFFiles();
|
||||
}
|
||||
|
||||
return this.storyStore.extract(options);
|
||||
}
|
||||
|
||||
// UTILITIES
|
||||
async cleanupPreviousRender({ unmountDocs = true }: { unmountDocs?: boolean } = {}) {
|
||||
const previousViewMode = this.previousStory?.parameters?.docsOnly
|
||||
? 'docs'
|
||||
@ -724,6 +752,7 @@ export class PreviewWeb<TFramework extends AnyFramework> {
|
||||
}
|
||||
|
||||
renderPreviewEntryError(reason: string, err: Error) {
|
||||
this.previewEntryError = err;
|
||||
logger.error(reason);
|
||||
logger.error(err);
|
||||
this.view.showErrorDisplay(err);
|
||||
|
Loading…
x
Reference in New Issue
Block a user