Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Enable support for React 18, bump Gatsby to v5 #3367

Open
wants to merge 37 commits into
base: release-22.x
Choose a base branch
from

Conversation

bradenmacdonald
Copy link
Contributor

@bradenmacdonald bradenmacdonald commented Dec 16, 2024

Description

We are long overdue for supporting React 18. The challenge is that bumping to React 18 and testing with it on our docs site also requires bumping Gatsby from v4 to v5 and gatsby-plugin-mdx from v3 to v5, which introduces a ton of breaking changes to how Gatsby and MDX work.

These upgrades have been attempted before in #2774 and #2767 but so far nobody has been able to finish the work up. I'm thinking we need to bite the bullet sooner or later and get it sort out.

Deploy Preview

https://deploy-preview-3367--paragon-openedx-v22.netlify.app/

Note: Compare to https://paragon-openedx-v22.netlify.app , not to https://paragon-openedx.netlify.app/ !

Status

Working:

  • Paragon build (npm run build)
  • Paragon in general (the components on the docs site work as normal - see preview above)
  • The docs site (except for some minor issues; see below)
  • Test suite
  • Lint checks
  • Fully backwards compatible (still supports React 16 & 17)

Issues and solutions

  • Need to test this with an existing MFE to confirm backwards compatibility
  • Modals seem to be throwing a JS error on the docs site and the useIsVisible hook (link) doesn't seem to be working properly anymore.
  • The icons are showing a lot of babel compatibility boilerplate again, i.e. the changes from Reduce dist size by 500 KB - reinstate browserslist, removing transforms for ancient browsers #3284 seem to be undone after upgrading to React 18.
  • The home page of the docs site lists pages like "Icons" in the Components list ("Layout" appears twice - once as a component and once as a page).
  • The "Components" part of the "Foundations/Layout" page uses a jsx live area that is just appearing as plain code, not rendered components.
  • Components like Button are missing "Theme Variables (SCSS)" - compare new Button page vs. existing v22 Button page
  • The "Playground" page is not working.
  • Horizontal padding, font sizes, and other minor changes are seen on pages like the Button component docs.
    • Turns out this is from comparing v22 to v23. Lots of minor styling changes are visible on the v23 docs site.
  • Some of the "Foundations" pages (the ones that use MDX and pageQuery data - specifically "Layout" and "CSS Utilities" , have errors and aren't loading properly. Fixed with aed4ef4 and 8abf42c
  • Some of the "Foundations" pages (the ones that use MDX, not TSX, like Layout , are not rendering with the correct layout. Fixed by @brian-smith-tcril via e0a3539
  • Currently, gatsby is running out of memory when trying to run npm run start. I thought I had solved this with GATSBY_CPU_COUNT=4 (f67d1f7) but it no longer seems to be helping (even if lowered down to 1). I think the memory usage depends on how much has been cached, so clearing the cache always requires a lot more memory. For now the only workaround is to run it with NODE_OPTIONS=--max-old-space-size=16384 npm run start but allocating 16GB each time is not feasible.
    • Note that npm run build-docs seems to be working fine, and the "deploy preview" build of this PR works fine too.
    • Solved! By @brian-smith-tcril who found that adding DEV_SSR/FAST_DEV will avoid the issue.
  • This PR necessarily bumps the version of sass used via transitive dependencies. Will this version bump cause issues for Paragon consumers (if we change the sass code to the new version)? e.g. see 958f617
  • Upgrading sass means instead of e.g. map-get(...) we are encouraged to use @use "sass:map"; and map.get(...) - done via 958f617 but this may need to be reverted, see previous point.
  • Stylelint is not passing
  • Test cases need to be updated - done with e.g. c5ca0a5 , 1942f03 and be8e10a
  • Many other sass deprecation warnings related to import etc. have been suppressed for now: 6946d67 we probably can't address this properly without a major bootstrap upgrade
  • GraphQL syntax has changed in some cases, resulting in a bunch of deprecation warnings - solved with 2dd6935
  • The { and < characters must now be escaped in markdown - solved with 0f6004f
  • The childMdx API is no longer available - gatsby-plugin-mdx no longer allows rendering multiple MDX documents in one page. I worked around this by adding react-markdown and rendering the props descriptions as plain markdown (they don't use MDX anyways) - 48aa670
  • Some Gatsby/rehype plugins can only be imported as ESM - so I had to convert the Gatsby config to ESM 60e534c
  • Code blocks in Markdown no longer pass in attributes like live automatically, so I had to add rehype-mdx-code-props in 11b7605
  • Supporting <PropsTable> as an MDX component was tricky because it requires some context from the parent JSX component, which was being inlined as a local variable resulting in lint warnings: "Do not define components during render". It turns out that the props tables in question, on this Data Views page are already present in the usual place at the end of the document anyways. So I think it's more consistent to just remove support for this element from that page (remove the duplicate props tables) and avoid having to re-engineer PropsTable to better support context for this one very questionable use case. Done with c92a1a2
  • update imports for 'renderHook' (now part of @testing-library/react) - 53500ee

@openedx-webhooks
Copy link

openedx-webhooks commented Dec 16, 2024

Thanks for the pull request, @bradenmacdonald!

This repository is currently maintained by @openedx/paragon-working-group.

Once you've gone through the following steps feel free to tag them in a comment and let them know that your changes are ready for engineering review.

🔘 Get product approval

If you haven't already, check this list to see if your contribution needs to go through the product review process.

  • If it does, you'll need to submit a product proposal for your contribution, and have it reviewed by the Product Working Group.
    • This process (including the steps you'll need to take) is documented here.
  • If it doesn't, simply proceed with the next step.

🔘 Provide context

To help your reviewers and other members of the community understand the purpose and larger context of your changes, feel free to add as much of the following information to the PR description as you can:

  • Dependencies

    This PR must be merged before / after / at the same time as ...

  • Blockers

    This PR is waiting for OEP-1234 to be accepted.

  • Timeline information

    This PR must be merged by XX date because ...

  • Partner information

    This is for a course on edx.org.

  • Supporting documentation
  • Relevant Open edX discussion forum threads

🔘 Get a green build

If one or more checks are failing, continue working on your changes until this is no longer the case and your build turns green.


Where can I find more information?

If you'd like to get more details on all aspects of the review process for open source pull requests (OSPRs), check out the following resources:

When can I expect my changes to be merged?

Our goal is to get community contributions seen and reviewed as efficiently as possible.

However, the amount of time that it takes to review and merge a PR can vary significantly based on factors such as:

  • The size and impact of the changes that it introduces
  • The need for product review
  • Maintenance status of the parent repository

💡 As a result it may take up to several weeks or months to complete a review and merge your PR.

@openedx-webhooks openedx-webhooks added the open-source-contribution PR author is not from Axim or 2U label Dec 16, 2024
Copy link

netlify bot commented Dec 16, 2024

Deploy Preview for paragon-openedx-v23 ready!

Name Link
🔨 Latest commit c5ca0a5
🔍 Latest deploy log https://app.netlify.com/sites/paragon-openedx-v23/deploys/67608bbcdeb4ce0008d6f3bc
😎 Deploy Preview https://deploy-preview-3367--paragon-openedx-v23.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

@bradenmacdonald bradenmacdonald changed the base branch from release-23.x to master December 16, 2024 20:22
@bradenmacdonald bradenmacdonald changed the base branch from master to release-22.x December 16, 2024 22:22
@brian-smith-tcril
Copy link
Contributor

brian-smith-tcril commented Dec 17, 2024

After reading through a ton of gatsbyjs/gatsby#36899 and trying a few things I found gatsbyjs/gatsby#36899 (comment) and managed to get the dev server running.

bsmith@aximdev:~/code/paragon$ git diff
diff --git a/www/gatsby-config.mjs b/www/gatsby-config.mjs
index d0ab6c658e..aaf5daab39 100644
--- a/www/gatsby-config.mjs
+++ b/www/gatsby-config.mjs
@@ -130,4 +130,7 @@ export default {
   // Match the location of the site on github pages if no path prefix is specified
   pathPrefix: process.env.PATH_PREFIX || '',
   plugins,
+  flags: {
+    DEV_SSR: true
+  },
 };

Edit: I looked into this a bit more and found https://www.gatsbyjs.com/docs/reference/release-notes/v2.28/#feature-flags-in-gatsby-configjs. Setting the FAST_DEV flag also works (and seems like a good thing to do!)

@brian-smith-tcril
Copy link
Contributor

Re:

Some of the "Foundations" pages (the ones that use MDX, not TSX, like Layout , are not rendering with the correct layout. Some fix is required (see related work in #2767 which has some code related to this)

With this change

-        default: require.resolve(
-          './src/templates/default-mdx-page-template.tsx',
-        ),

It seems the default-mdx-page-template layout is just not being used at all. I know defaultLayouts were removed in v4, and the migration of component-page-template seems to have gone smoothly, but I'm a little lost as to how to properly migrate the other layout.

…rver

Replaces/reverts "fix: limit parallelization of build to avoid OOM error during 'develop'"
Copy link

netlify bot commented Dec 17, 2024

Deploy Preview for paragon-openedx-v22 ready!

Name Link
🔨 Latest commit 8979107
🔍 Latest deploy log https://app.netlify.com/sites/paragon-openedx-v22/deploys/6786b3003847a800081cf241
😎 Deploy Preview https://deploy-preview-3367--paragon-openedx-v22.netlify.app
📱 Preview on mobile
Toggle QR Code...

QR Code

Use your smartphone camera to open QR code link.

To edit notification comments on pull requests, go to your Netlify site configuration.

@bradenmacdonald
Copy link
Contributor Author

It seems the default-mdx-page-template layout is just not being used at all. I know defaultLayouts were removed in v4, and the migration of component-page-template seems to have gone smoothly, but I'm a little lost as to how to properly migrate the other layout.

Yeah, me too. I tried a couple things based on the docs but didn't have much luck yet. There is some perhaps complication from the fact that Gatsby includes everything in src/pages/ by default, but I've also added those pages explicitly using a gatsby-source-filesystem in gatsby-config.mjs because I was seeing some Could not locate File node errors if I didn't include that in the config (though it wasn't needed with the previous version).

Clearly, there is somewhere that we need to specify that default-mdx-page-template should be used, but I haven't found where yet, nor figured out why putting it in www/utils/createPages.js similarly to how #2767 did it isn't working.

@brian-smith-tcril
Copy link
Contributor

So I managed to get the layout page to render, I had to comment some stuff in the .mdx out and there are still a few errors, but it's at least a jumping off point

braden/react-gatsby-bump...5589454

@bradenmacdonald
Copy link
Contributor Author

@brian-smith-tcril Nice! I pulled in most of that change for now. It seems like pages like icons.mdx are now working perfectly, so the layout part is correct - just need to solve the random errors on the other pages.

@bradenmacdonald
Copy link
Contributor Author

bradenmacdonald commented Dec 18, 2024

For some reason, either when I changed the target branch to release-22.x or when I pushed that fix from @brian-smith-tcril, the "v22" deploy preview is now broken and can't load the theme CSS: https://deploy-preview-3367--paragon-openedx-v22.netlify.app/

@brian-smith-tcril
Copy link
Contributor

@bradenmacdonald the build step of the latest deploy of this branch on netlify failed. I pulled the latest version of this branch and tried running npm run build-docs and it failed in the same place locally for me.

I commented out a couple things in layout.mdx and css-utilities.mdx and got it to build brian-smith-tcril@e42915b

@bradenmacdonald
Copy link
Contributor Author

Ah, thanks @brian-smith-tcril :) I'll hold off on committing that for now until I see if we can actually get that < CSSUtilitiesTable> component working.

@brian-smith-tcril
Copy link
Contributor

re:

  • This PR necessarily bumps the version of sass used via transitive dependencies. Will this version bump cause issues for Paragon consumers (if we change the sass code to the new version)? e.g. see 958f617
    • ‼️ help answering this would be appreciated ‼️

We discussed this in the WG meeting yesterday. It seems the version bump is a minor version bump as opposed to a major one:

  • Since it's a minor bump, that means existing scss in MFEs should still work as expected
  • It seems using new features that don't exist in earlier minor versions of sass in Paragon also shouldn't be an issue. MFEs that upgrade to the new version of Paragon that has the new version of sass should automatically get the new version of sass in their package locks.

If there's some nuance here I'm missing please let me know!

@bradenmacdonald
Copy link
Contributor Author

@brian-smith-tcril FYI, I got the <CSSUtilitiesTable> working now, so the build/preview is back:

https://deploy-preview-3367--paragon-openedx-v22.netlify.app/

@bradenmacdonald bradenmacdonald force-pushed the braden/react-gatsby-bump branch from c85a647 to 7fbf83e Compare January 9, 2025 20:26
@bradenmacdonald
Copy link
Contributor Author

@brian-smith-tcril When I pulled in the changes from #3125 into this branch, I encountered the same problem with webpack confusing two files like useToggle.tsx and useToggle.mdx. It seems that your workaround of explicitly adding gatsby-plugin-typescript doesn't work anymore in Gatsby 5 (well actually, it does work, but it breaks the MDX plugin).

So what I've done for now is renamed e.g. useToggle.tsx to useToggleHook.tsx (as opposed to your suggestion of renaming useToggle.mdx to useToggleDoc.mdx). This should maintain both API compatibility and keep the URLs unchanged, unless I'm missing something. Does that make sense?

@brian-smith-tcril
Copy link
Contributor

So what I've done for now is renamed e.g. useToggle.tsx to useToggleHook.tsx (as opposed to your suggestion of renaming useToggle.mdx to useToggleDoc.mdx). This should maintain both API compatibility and keep the URLs unchanged, unless I'm missing something. Does that make sense?

That sounds like a very sensible solution to me!

Copy link

codecov bot commented Jan 9, 2025

Codecov Report

All modified and coverable lines are covered by tests ✅

Project coverage is 93.44%. Comparing base (b2da298) to head (8979107).

Additional details and impacted files
@@               Coverage Diff                @@
##           release-22.x    #3367      +/-   ##
================================================
- Coverage         93.49%   93.44%   -0.05%     
================================================
  Files               252      251       -1     
  Lines              4655     4519     -136     
  Branches           1088     1016      -72     
================================================
- Hits               4352     4223     -129     
+ Misses              296      293       -3     
+ Partials              7        3       -4     

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

@brian-smith-tcril
Copy link
Contributor

@bradenmacdonald I see there are only a couple things left unchecked:

I'd be more than happy to dive in and try to figure some of those out - is there one in particular you'd like another set of eyes on?

@bradenmacdonald
Copy link
Contributor Author

@brian-smith-tcril I don't know much about the Playground page, so that would probably be the one I could most use help with - thanks!

Copy link
Member

@adamstankiewicz adamstankiewicz left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for tackling these changes, @bradenmacdonald! Left a few comments throughout. Here's a couple other issues I noticed when comparing this PR's deploy preview vs. the v22 docs site:

  1. Modals seem to be throwing a JS error on the docs site when trying to open them. For example, see AlertModal (link) and click the "Open alert modal" button. Observe a JS error TypeError: object is not iterable (cannot read property Symbol(Symbol.iterator)). This seems to occur on each modal variant that extends ModalDialog.
  2. The useIsVisible hook (link) doesn't seem to be working properly anymore. The "Is element visible? yes/no" in the live code example is not reflecting whether the element is actually visible or not (consistently says "no" even when the element is visible).

@@ -17,14 +17,10 @@
"license": "ISC",
"dependencies": {
"@edx/brand": "npm:@openedx/brand-openedx@^1.2.2",
"@edx/frontend-platform": "^4.2.0",
"@edx/frontend-build": "^13.0.14",
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[clarification] Why is @edx/frontend-build moved to a regular dependency vs. keeping it as a dev dependency?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is for the example MFE (not Paragon itself), and I'm following the best practice I established for MFEs that "dependencies" are what you need to build it (before deploying), and "devDependencies" are for development-only things like tests and linting.

If we had a part of the MFE that runs on the server, it would be different, but there's no server-side code here.

Comment on lines +138 to +141
if (className === '' && typeof children === 'string' && !children.includes('\n')) {
// This is an inline code block. Don't use syntax highlighting.
return <code>{children}</code>;
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[curious] When/where does this conditional for an inline code block get used?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Anywhere you have an `inline code node` like this:

Screenshot

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(I shouldn't have used the word "block")

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah, I see. Makes sense, though somewhat odd that on master, the CodeBlock was not previously used for these inline code nodes, despite having code: CodeBlock configured within the previous implementation's shortCodes (source). For example, adding a

With CodeBlock actually being used now for these inline code nodes, makes sense we'd need to handle it explicitly. Wonder why it wasn't used before. Either way, thanks for clarifying!

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, I am also wondering why. Those inline nodes are still using code as their tag before this PR, so it's unclear to me why they weren't using CodeBlock. In any case, it was an easy workaround.

www/src/components/LinkedHeading.tsx Outdated Show resolved Hide resolved
Comment on lines -38 to +46
sort: { fields: frontmatter___title }
sort: { frontmatter: {title: ASC} }
) {
categories: group(field: frontmatter___categories) {
categories: group(field: {frontmatter: {categories: SELECT}}) {
nodes {
...ComponentPage
}
fieldValue
}
types: group(field: frontmatter___type) {
types: group(field: {frontmatter: {type: SELECT}}) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure if these are the specific changes causing the change, but I noticed on the PR deploy preview contains additional links in the footer menu compared to the v22 docs site.

The PR deploy preview seems to include static content pages instead of the intended exports (i.e., components, hooks, etc.). For example, the footer menu now contains things like "Brand-Icons", "Component generation", "Getting started", etc.).

Is this intentional?

Also worth noting, clicking at least the "CSS Utilities" in the footer menu leads to a 404 not found page (link).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, those additional links are a bug and I had already noted them in the list of issues above that I need to work through. I did not intend to make any changes to the links (or really any visual/content changes to the docs site at all - any that you see are a bug).

@@ -53,7 +55,7 @@
margin: 6.875rem auto auto;
cursor: auto;

@media (max-width: map-get($grid-breakpoints, "md")) {
@media (max-width: map.get($grid-breakpoints, "md")) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

[curious] I noticed some @media queries were modified to replace things like max-width with width >= (e.g., @media only screen and (width <= 768px) { in Toast).

  1. Should we be consistent about using max-width and min-width vs. just width in our media queries?
  2. Should any hardcoded pixel values in media queries (like the Toast example above) be using the $grid-breakpoints via map.get?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I noticed some @media queries were modified to replace things like max-width with width >= (e.g., @media only screen and (width <= 768px) { in Toast).

Those changes were done automatically when I ran sass-migrator on the codebase (though I ended up committing only a subset of the various changes it made). So I assumed that they're equivalent syntax, that this way is the new best practice, and that linters [will] warn about the previous way. But I didn't check those assumptions or look into it in much detail. The width <= 768px seems simple and reasonable to me. That change isn't necessary as far as I know though, so let me know if you see any issues.

Should any hardcoded pixel values in media queries (like the Toast example above) be using the $grid-breakpoints via map.get?

Probably, but let's do that in a follow-up PR as this one is already huge, and I'm just trying to do the bare minimum necessary for compatibility with the new React/Sass/Gatsby versions.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The width <= 768px seems simple and reasonable to me. That change isn't necessary as far as I know though, so let me know if you see any issues.

No issues immediately observed; I agree the change should be functionally equivalent. The newer syntax is easier definitely easier to grok.

Probably, but let's do that in a follow-up PR as this one is already huge

Totally, just wanted to call it out for now :)

www/utils/createPages.js Outdated Show resolved Hide resolved
@bradenmacdonald
Copy link
Contributor Author

  • Modals seem to be throwing a JS error on the docs site when trying to open them. For example, see AlertModal (link) and click the "Open alert modal" button. Observe a JS error TypeError: object is not iterable (cannot read property Symbol(Symbol.iterator)). This seems to occur on each modal variant that extends ModalDialog.
  • The useIsVisible hook (link) doesn't seem to be working properly anymore. The "Is element visible? yes/no" in the live code example is not reflecting whether the element is actually visible or not (consistently says "no" even when the element is visible).

Ah, these were actually two more symptoms of the same general problem of webpack getting confused between .mdx and .tsx files with the same name. I should have been more thorough in my earlier fix. In any case, it's fixed now with 8979107

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
open-source-contribution PR author is not from Axim or 2U
Projects
Status: In Eng Review
Development

Successfully merging this pull request may close these issues.

4 participants