From e0d9f35868265de1aad74b001daec1cab8f8e769 Mon Sep 17 00:00:00 2001 From: RheeseyB <1044774+Rheeseyb@users.noreply.github.com> Date: Fri, 24 Nov 2023 10:36:33 +0000 Subject: [PATCH] Prevent cyclic dependencies in remix projects (#4557) * fix(editor) Prevent cyclic dependencies in remix projects * chore(editor) Added a test for remix cyclic dependencies --- .../remix/remix-rendering.spec.browser2.tsx | 20 +++++++++++++++++-- .../components/canvas/remix/remix-utils.tsx | 9 +++++---- 2 files changed, 23 insertions(+), 6 deletions(-) diff --git a/editor/src/components/canvas/remix/remix-rendering.spec.browser2.tsx b/editor/src/components/canvas/remix/remix-rendering.spec.browser2.tsx index 06e4dfd41c11..a36401e19ba1 100644 --- a/editor/src/components/canvas/remix/remix-rendering.spec.browser2.tsx +++ b/editor/src/components/canvas/remix/remix-rendering.spec.browser2.tsx @@ -70,10 +70,15 @@ async function renderRemixProject(project: PersistentModel) { } describe('Remix content', () => { - it('Renders the remix container with actual content', async () => { + it('Renders the remix container with actual content and a cyclic dependency', async () => { const project = createModifiedProject({ [StoryboardFilePath]: `import * as React from 'react' import { RemixScene, Storyboard } from 'utopia-api' + import { Card } from '/app/components/card' + + export function gimmeData() { + return '${DefaultRouteTextContent}' + } export var storyboard = ( @@ -90,6 +95,16 @@ describe('Remix content', () => { ) `, + ['/app/components/card.js']: `import * as React from 'react' + import { gimmeData } from '${StoryboardFilePath}' + + export const Card = (props) => { + const data = gimmeData() + return ( +

{data}

+ ) + } + `, ['/app/root.js']: `import React from 'react' import { Outlet } from '@remix-run/react' @@ -103,9 +118,10 @@ describe('Remix content', () => { } `, ['/app/routes/_index.js']: `import React from 'react' + import { Card } from '/app/components/card' export default function Index() { - return

${DefaultRouteTextContent}

+ return } `, }) diff --git a/editor/src/components/canvas/remix/remix-utils.tsx b/editor/src/components/canvas/remix/remix-utils.tsx index 49462aa8ff28..39d45bb82987 100644 --- a/editor/src/components/canvas/remix/remix-utils.tsx +++ b/editor/src/components/canvas/remix/remix-utils.tsx @@ -255,7 +255,7 @@ function getRemixExportsOfModule( displayNoneInstances: Array, metadataContext: UiJsxCanvasContextData, ) => { - let resolvedFiles: MapLike> = {} + let resolvedFiles: MapLike> = {} let resolvedFileNames: Array = [filename] const requireFn = curriedRequireFn(innerProjectContents) @@ -267,11 +267,12 @@ function getRemixExportsOfModule( } let resolvedFromThisOrigin = resolvedFiles[importOrigin] - const alreadyResolved = resolvedFromThisOrigin.includes(toImport) // We're inside a cyclic dependency, so trigger the below fallback const filePathResolveResult = alreadyResolved + const alreadyResolved = resolvedFromThisOrigin[toImport] !== undefined + const filePathResolveResult = alreadyResolved ? left('Already resolved') : resolve(importOrigin, toImport) - forEachRight(alreadyResolved, (filepath) => resolvedFileNames.push(filepath)) + forEachRight(filePathResolveResult, (filepath) => resolvedFileNames.push(filepath)) const resolvedParseSuccess: Either> = attemptToResolveParsedComponents( resolvedFromThisOrigin, @@ -287,7 +288,7 @@ function getRemixExportsOfModule( metadataContext, NO_OP, false, - alreadyResolved, + filePathResolveResult, null, ) return foldEither(