Don't crash when there are errors indexing

This commit is contained in:
Tom Coleman 2023-02-16 15:36:41 +11:00
parent 23d7fd2da7
commit 1b99fc225d
4 changed files with 66 additions and 32 deletions

View File

@ -956,7 +956,8 @@ describe('StoryIndexGenerator', () => {
[normalizeStoriesEntry('./src/docs2/MetaOf.mdx', options)],
options
);
await expect(() => generator.initialize()).rejects.toThrowError(
await generator.initialize();
await expect(() => generator.getIndex()).rejects.toThrowError(
/Could not find "..\/A.stories" for docs file/
);
});

View File

@ -35,7 +35,11 @@ type StoriesCacheEntry = {
dependents: Path[];
type: 'stories';
};
type CacheEntry = false | StoriesCacheEntry | DocsCacheEntry;
type ErrorEntry = {
type: 'error';
err: Error;
};
type CacheEntry = false | StoriesCacheEntry | DocsCacheEntry | ErrorEntry;
type SpecifierStoriesCache = Record<Path, CacheEntry>;
export const AUTODOCS_TAG = 'autodocs';
@ -165,7 +169,12 @@ export class StoryIndexGenerator {
return Promise.all(
Object.keys(entry).map(async (absolutePath) => {
if (entry[absolutePath] && !overwrite) return;
try {
entry[absolutePath] = await updater(specifier, absolutePath, entry[absolutePath]);
} catch (err) {
entry[absolutePath] = { type: 'error', err };
}
})
);
})
@ -176,7 +185,7 @@ export class StoryIndexGenerator {
return /(?<!\.stories)\.mdx$/i.test(absolutePath);
}
async ensureExtracted(): Promise<IndexEntry[]> {
async ensureExtracted(): Promise<(IndexEntry | ErrorEntry)[]> {
// First process all the story files. Then, in a second pass,
// process the docs files. The reason for this is that the docs
// files may use the `<Meta of={XStories} />` syntax, which requires
@ -193,9 +202,10 @@ export class StoryIndexGenerator {
return this.specifiers.flatMap((specifier) => {
const cache = this.specifierToCache.get(specifier);
return Object.values(cache).flatMap((entry): IndexEntry[] => {
return Object.values(cache).flatMap((entry): (IndexEntry | ErrorEntry)[] => {
if (!entry) return [];
if (entry.type === 'docs') return [entry];
if (entry.type === 'error') return [entry];
return entry.entries;
});
});
@ -481,7 +491,11 @@ export class StoryIndexGenerator {
// Extract any entries that are currently missing
// Pull out each file's stories into a list of stories, to be composed and sorted
const storiesList = await this.ensureExtracted();
const sorted = await this.sortStories(storiesList);
const firstError = storiesList.find((entry) => entry.type === 'error');
if (firstError) throw (firstError as ErrorEntry).err;
const sorted = await this.sortStories(storiesList as IndexEntry[]);
let compat = sorted;
if (this.options.storiesV2Compatibility) {

View File

@ -1,10 +1,11 @@
import type { CoreConfig, Options } from '@storybook/types';
import type { CoreConfig, Options, StoryIndex } from '@storybook/types';
import { telemetry, getPrecedingUpgrade } from '@storybook/telemetry';
import { useStorybookMetadata } from './metadata';
import type { StoryIndexGenerator } from './StoryIndexGenerator';
import { summarizeIndex } from './summarizeIndex';
import { router } from './router';
import { versionStatus } from './versionStatus';
import { sendTelemetryError } from '../withTelemetry';
export async function doTelemetry(
core: CoreConfig,
@ -13,7 +14,18 @@ export async function doTelemetry(
) {
if (!core?.disableTelemetry) {
initializedStoryIndexGenerator.then(async (generator) => {
const storyIndex = await generator?.getIndex();
let storyIndex: StoryIndex;
try {
storyIndex = await generator?.getIndex();
} catch (err) {
// If we fail to get the index, treat it as a recoverable error, but send it up to telemetry
// as if we crashed. In the future we will revisit this to send a distinct error
sendTelemetryError(err, 'dev', {
cliOptions: options,
presetOptions: { ...options, corePresets: [], overridePresets: [] },
});
return;
}
const { versionCheck, versionUpdates } = options;
const payload = {
precedingUpgrade: await getPrecedingUpgrade(),

View File

@ -58,17 +58,11 @@ async function getErrorLevel({ cliOptions, presetOptions }: TelemetryOptions): P
return 'full';
}
export async function withTelemetry(
export async function sendTelemetryError(
error: Error,
eventType: EventType,
options: TelemetryOptions,
run: () => Promise<any>
options: TelemetryOptions
) {
if (!options.cliOptions.disableTelemetry)
telemetry('boot', { eventType }, { stripMetadata: true });
try {
await run();
} catch (error) {
try {
const errorLevel = await getErrorLevel(options);
if (errorLevel !== 'none') {
@ -92,7 +86,20 @@ export async function withTelemetry(
} catch (err) {
// if this throws an error, we just move on
}
}
export async function withTelemetry(
eventType: EventType,
options: TelemetryOptions,
run: () => Promise<any>
) {
if (!options.cliOptions.disableTelemetry)
telemetry('boot', { eventType }, { stripMetadata: true });
try {
await run();
} catch (error) {
await sendTelemetryError(error, eventType, options);
throw error;
}
}