From 33b487f1e71bb91382c3842397cfe35e1dc62d41 Mon Sep 17 00:00:00 2001 From: Lars Rickert Date: Sat, 13 Jul 2024 18:52:26 +0200 Subject: [PATCH 1/4] fix(vue-component-meta): fix out of memory issue --- .../src/plugins/vue-component-meta.ts | 19 +++++++++++++++++++ .../vue3/src/docs/extractArgTypes.ts | 12 +++--------- 2 files changed, 22 insertions(+), 9 deletions(-) diff --git a/code/frameworks/vue3-vite/src/plugins/vue-component-meta.ts b/code/frameworks/vue3-vite/src/plugins/vue-component-meta.ts index 54ac30cbb9dd..d5d3e96bfa64 100644 --- a/code/frameworks/vue3-vite/src/plugins/vue-component-meta.ts +++ b/code/frameworks/vue3-vite/src/plugins/vue-component-meta.ts @@ -50,6 +50,25 @@ export async function vueComponentMeta(tsconfigPath = 'tsconfig.json'): Promise< const exportName = exportNames[index]; + // we remove nested object schemas here since they are not used inside Storybook (we don't generate controls for object properties) + // and they can cause "out of memory" issues for large/complex schemas (e.g. HTMLElement) + // it also reduced the bundle size when running "Storybook build" when such schemas are used + (['props', 'exposed'] as const).forEach((key) => { + meta[key].forEach((value) => { + if (typeof value.schema !== 'object') return; + + // we need to use Object.defineProperty here since schema is a getter so we can not set it directly + Object.defineProperty(value, 'schema', { + configurable: true, + enumerable: true, + value: { + kind: value.schema.kind, + type: value.schema.type, + }, + }); + }); + }); + const exposed = // the meta also includes duplicated entries in the "exposed" array with "on" // prefix (e.g. onClick instead of click), so we need to filter them out here diff --git a/code/renderers/vue3/src/docs/extractArgTypes.ts b/code/renderers/vue3/src/docs/extractArgTypes.ts index 76c52cc755a7..164f5957e693 100644 --- a/code/renderers/vue3/src/docs/extractArgTypes.ts +++ b/code/renderers/vue3/src/docs/extractArgTypes.ts @@ -1,3 +1,4 @@ +import type { VueDocgenInfo, VueDocgenInfoEntry, VueDocgenPlugin } from '@storybook/vue3-vite'; import type { ExtractedProp } from 'storybook/internal/docs-tools'; import { convert, @@ -6,7 +7,6 @@ import { type ArgTypesExtractor, } from 'storybook/internal/docs-tools'; import type { SBType, StrictArgTypes, StrictInputType } from 'storybook/internal/types'; -import type { VueDocgenInfo, VueDocgenInfoEntry, VueDocgenPlugin } from '@storybook/vue3-vite'; type PropertyMetaSchema = VueDocgenInfoEntry<'vue-component-meta', 'props'>['schema']; @@ -283,17 +283,11 @@ export const convertVueComponentMetaProp = ( }; } - // recursively/deeply convert all properties of the object case 'object': return { name: 'object', - value: Object.entries(schema.schema ?? {}).reduce>( - (obj, [propName, propSchema]) => { - obj[propName] = convertVueComponentMetaProp(propSchema); - return obj; - }, - {} - ), + // Storybook does not generate controls for object properties so we don't need to recursively map the object schema here + value: {}, required, }; From 70a6a41b0fdb504f2c79ed8ab49038bb187f122c Mon Sep 17 00:00:00 2001 From: Lars Rickert Date: Sat, 13 Jul 2024 19:08:59 +0200 Subject: [PATCH 2/4] update comment --- code/frameworks/vue3-vite/src/plugins/vue-component-meta.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/code/frameworks/vue3-vite/src/plugins/vue-component-meta.ts b/code/frameworks/vue3-vite/src/plugins/vue-component-meta.ts index d5d3e96bfa64..a263635a44f8 100644 --- a/code/frameworks/vue3-vite/src/plugins/vue-component-meta.ts +++ b/code/frameworks/vue3-vite/src/plugins/vue-component-meta.ts @@ -64,6 +64,7 @@ export async function vueComponentMeta(tsconfigPath = 'tsconfig.json'): Promise< value: { kind: value.schema.kind, type: value.schema.type, + // note that value.schema.schema is not included here (see comment above) }, }); }); From ebe47fff6e842b2c5a92b0c15a7045b3e3ec63df Mon Sep 17 00:00:00 2001 From: Lars Rickert Date: Sat, 13 Jul 2024 20:35:20 +0200 Subject: [PATCH 3/4] fix test --- .../extractArgTypes.test.ts.snap | 61 +++---------------- 1 file changed, 8 insertions(+), 53 deletions(-) diff --git a/code/renderers/vue3/src/docs/__snapshots__/extractArgTypes.test.ts.snap b/code/renderers/vue3/src/docs/__snapshots__/extractArgTypes.test.ts.snap index cbe74a47b7d6..39e2654bcff4 100644 --- a/code/renderers/vue3/src/docs/__snapshots__/extractArgTypes.test.ts.snap +++ b/code/renderers/vue3/src/docs/__snapshots__/extractArgTypes.test.ts.snap @@ -157,12 +157,7 @@ exports[`extractArgTypes (vue-docgen-api) > should extract props for component 1 "value": { "name": "object", "required": false, - "value": { - "nestedProp": { - "name": "string", - "required": true, - }, - }, + "value": {}, }, }, }, @@ -183,12 +178,7 @@ exports[`extractArgTypes (vue-docgen-api) > should extract props for component 1 "value": { "name": "object", "required": false, - "value": { - "nestedProp": { - "name": "string", - "required": true, - }, - }, + "value": {}, }, }, }, @@ -279,12 +269,7 @@ exports[`extractArgTypes (vue-docgen-api) > should extract props for component 1 "type": { "name": "object", "required": true, - "value": { - "foo": { - "name": "string", - "required": true, - }, - }, + "value": {}, }, }, "literalFromContext": { @@ -325,12 +310,7 @@ exports[`extractArgTypes (vue-docgen-api) > should extract props for component 1 "type": { "name": "object", "required": true, - "value": { - "nestedProp": { - "name": "string", - "required": true, - }, - }, + "value": {}, }, }, "nestedIntersection": { @@ -347,16 +327,7 @@ exports[`extractArgTypes (vue-docgen-api) > should extract props for component 1 "type": { "name": "object", "required": true, - "value": { - "additionalProp": { - "name": "string", - "required": true, - }, - "nestedProp": { - "name": "string", - "required": true, - }, - }, + "value": {}, }, }, "nestedOptional": { @@ -377,22 +348,12 @@ exports[`extractArgTypes (vue-docgen-api) > should extract props for component 1 { "name": "object", "required": false, - "value": { - "nestedProp": { - "name": "string", - "required": true, - }, - }, + "value": {}, }, { "name": "object", "required": false, - "value": { - "nestedProp": { - "name": "string", - "required": true, - }, - }, + "value": {}, }, ], }, @@ -411,13 +372,7 @@ exports[`extractArgTypes (vue-docgen-api) > should extract props for component 1 "type": { "name": "object", "required": false, - "value": { - "recursive": { - "name": "other", - "required": true, - "value": "MyNestedRecursiveProps", - }, - }, + "value": {}, }, }, "stringArray": { From e75ac8b4da8b838e45731abb073322166624df0c Mon Sep 17 00:00:00 2001 From: Lars Rickert Date: Tue, 16 Jul 2024 20:44:05 +0200 Subject: [PATCH 4/4] update docs --- code/renderers/vue3/src/docs/extractArgTypes.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/code/renderers/vue3/src/docs/extractArgTypes.ts b/code/renderers/vue3/src/docs/extractArgTypes.ts index 164f5957e693..31f4d19dc163 100644 --- a/code/renderers/vue3/src/docs/extractArgTypes.ts +++ b/code/renderers/vue3/src/docs/extractArgTypes.ts @@ -286,7 +286,8 @@ export const convertVueComponentMetaProp = ( case 'object': return { name: 'object', - // Storybook does not generate controls for object properties so we don't need to recursively map the object schema here + // while Storybook generates simple JSON object controls, nested schemas don't have specialized controls + // so we don't need to recursively map the object schema here value: {}, required, };