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

Replace Webpacker with jsbundling-rails + Webpack #5746

Merged
merged 42 commits into from
Jan 12, 2022
Merged

Conversation

aduth
Copy link
Member

@aduth aduth commented Dec 22, 2021

Related: #5725

Why: To improve performance of JavaScript builds, to adopt modern defaults for Rails, and to migrate away from the now >1 year out-of-date Webpack 4 tied to the latest stable version of Webpacker (and in so doing, eliminate such version tying altogether).

Observations:

Good:

  • Build seems anecdotally faster
  • Doesn't feel like a drastic departure
    • We already extensively customized the Webpack configuration anyways
    • Our handling of localization scripts already necessitated manual manifest lookups, so we already had most of the logic in ScriptHelper to do this part ourselves
  • Drops a lot of unused dependencies from Webpacker, and makes our Webpack dependencies more explicit
    • This is also improved in Webpacker v6 by making most dependencies opt-in. But at this rate, I'm not sure if v6 will ever be released
  • More flexibility for Webpack configuration
    • Specifically wondering if swapping from babel-loader to swc-loader could be a good move to further improve build performance

Not-so-good:

  • The Webpack build itself is reasonably fast, but the Rails side of it can be rather slow, at least on initial boot-up Edit: This was addressed by bypassing Sprockets and outputting Webpack bundles directly to public.
  • Webpacker was better integrated into Rails requests and had some nice quality-of-life features we'd lose: Edit: This was addressed by running webpack-dev-server in parallel to the Rails process, and loading scripts as served from this dev server.
    • Automatic refresh on JavaScript file changes
    • Waiting for the Webpack compilation to finish before letting the response be presented
  • rails s will no longer include JavaScript compilation. You'd also need to run yarn build --watch in the background

Next:

A few of the included changes are things we could probably consider doing sooner than later, regardless of jsbundling-rails:

  • Stop compiling SASS with Webpack Edit: This was done separately in Standardize SASS processing to Sprockets #5793
  • Prepare @18f/identity-i18n-webpack-plugin for Webpack 5
    • Webpack v5 uses Set in many places where they are arrays in Webpack v4. This causes deprecation warnings. In most cases, we could cast those values to arrays ([...val]) which would work for either a set or an array.

app/helpers/script_helper.rb Outdated Show resolved Hide resolved
package.json Outdated Show resolved Hide resolved
@zachmargolis
Copy link
Contributor

Looked like CI was upset that webpacker.yml was removed so I pushed 3683f08 to handle that

@aduth
Copy link
Member Author

aduth commented Dec 22, 2021

Couple more thoughts:

  • We could avoid the Sprockets fingerprinting overhead by bypassing Sprockets altogether and dumping the output to public.
    • Basically what Webpacker does.
  • On this thought and on closer inspection, I'm not really sure that jsbundling-rails does a whole lot for us beyond providing a nice installation experience and giving a few Rake tasks.
    • May just be able to remove it.
  • At that point, we've basically just rolled our own Webpacker.
    • I'm not totally opposed to this.
    • One extra consideration is that Sprockets and Webpacker also generate GZIP/Brotli variants for production compilation. However, I don't think we're currently using this, so it's basically just a waste of resources anyways. This may change after 18F/identity-devops#3486 .

@zachmargolis
Copy link
Contributor

We could avoid the Sprockets fingerprinting overhead by bypassing Sprockets altogether and dumping the output to public.

I like that the final assets have fingerprints in them, because then it works well when we eventually deploy to a CDN and don't have to deal with asset breakages across deploys, would removing that remove some intermediate step? Or remove fingerprinting entirely?

@aduth
Copy link
Member Author

aduth commented Dec 22, 2021

I like that the final assets have fingerprints in them, because then it works well when we eventually deploy to a CDN and don't have to deal with asset breakages across deploys, would removing that remove some intermediate step? Or remove fingerprinting entirely?

Ah, sorry, I should have clarified: I would still aim to keep the fingerprint, but generate those in Webpack instead, using the [contenthash] template tag. I assume this would also have some impact on performance, though ideally not quite as noticeable as it is currently with Sprockets.

One thing I noticed in perusing Webpacker source is that they include the digest only in production environments, following some Webpack build performance recommendations.

@zachmargolis
Copy link
Contributor

  • On this thought and on closer inspection, I'm not really sure that jsbundling-rails does a whole lot for us beyond providing a nice installation experience and giving a few Rake tasks.

    • May just be able to remove it.

I feel like with any of these projects, it's hard to tell what will stay up to date and what will drift and get stale over time. If it looks like jsbundling-rails is up to date for now, we may as well keep it, this PR seems easy enough so if we did want to just remove it and do something more direct, we could do that too? One PR doesn't bind us to a library forever

@aduth
Copy link
Member Author

aduth commented Dec 22, 2021

Yeah, I also forgot it handles the assets:precompile bit, which is nice enough to defer.

@aduth
Copy link
Member Author

aduth commented Dec 23, 2021

Webpacker was better integrated into Rails requests and had some nice quality-of-life features we'd lose:

  • Automatic refresh on JavaScript file changes
  • Waiting for the Webpack compilation to finish before letting the response be presented

Restored the first of these in 72e1058.

I'm thinking the second could be possible as well, by one of either:

  • Do manifest lookup by HTTP request to devserver's version of the manifest, assuming the dev server would hold the request 'til compilation is complete.
  • Similar, in manifest lookup, make an HTTP request to a custom dev-server API which implements a long-polling endpoint, released when websocket result indicates compilation is complete.

app/helpers/script_helper.rb Show resolved Hide resolved
webpack.config.js Outdated Show resolved Hide resolved
@aduth
Copy link
Member Author

aduth commented Dec 23, 2021

Build seems anecdotally faster

  • Still need to measure

One data point: The assets:precompile phase of the javascript_build CI task takes 32s on this branch, compared to 1m35s on a different, recent branch.

Before: https://app.circleci.com/pipelines/github/18F/identity-idp/64363/workflows/796fe5bb-ffdc-4adb-9a38-1dc55f270cef/jobs/100173
After: https://app.circleci.com/pipelines/github/18F/identity-idp/64455/workflows/f635608a-ac76-45af-a6ce-4755f9e12c3d/jobs/100309

Edit: Also worth reiterating that this opens up the opportunity to switch from Babel to swc-loader. Locally, that reduces time of yarn build by a further 75%, from 8.2s to 2.1s.

webpack.config.js Outdated Show resolved Hide resolved
@aduth
Copy link
Member Author

aduth commented Dec 23, 2021

  • Do manifest lookup by HTTP request to devserver's version of the manifest, assuming the dev server would hold the request 'til compilation is complete.

Solid lead from logging:

[webpack-dev-middleware] wait until bundle finished: /packs/assets-manifest.json

@aduth
Copy link
Member Author

aduth commented Dec 23, 2021

Waiting for the Webpack compilation to finish before letting the response be presented

Actually... this may just work automatically, at least in these sense that:

  1. We don't digest generated packs in development
  2. Even if Rails were to use a stale manifest, the generated pack URLs are going to be the same
  3. Those pack URLs will be referencing the webpack-dev-server URL
  4. The webpack-dev-server will wait 'til any pending compilation is complete before serving them up

So even if compilation is in-progress while making the request, there would never be a case where the user would see behavior from the "old" code after refreshing.

One edge-case is if the manifest doesn't yet exist, such as when starting up the server for the first time. In my experience, the build takes less time to become ready than the Rails server. Alternatively, we could omit the manifest file from the clean script so that it continues to exist when starting the server, even if "stale".

Generally, it'd still be a good idea to have some safeguards hereabouts for if the file doesn't exist:

def manifest
@manifest ||= JSON.parse(File.read(MANIFEST_PATH))
end

@aduth
Copy link
Member Author

aduth commented Jan 3, 2022

Note to self: Do a pass on interoperability with devops workflows, specifically assumptions around Webpacker, public/packs, etc (thinking CloudFront in particular). I see at least a few references in identity-devops to public/packs and public/packs/manifest.json. They may just work as-is, but needs a closer look.

Currently, this branch flattens public/packs and doesn't have a nested js directory. For better compatibility and to anticipate a similar adoption of cssbundling-rails, it may be worth reintroducing it.

@mitchellhenke
Copy link
Contributor

Note to self: Do a pass on interoperability with devops workflows, specifically assumptions around Webpacker, public/packs, etc (thinking CloudFront in particular). I see at least a few references in identity-devops to public/packs and public/packs/manifest.json. They may just work as-is, but needs a closer look.

Currently, this branch flattens public/packs and doesn't have a nested js directory. For better compatibility and to anticipate a similar adoption of cssbundling-rails, it may be worth reintroducing it.

Similar assumptions exist in CircleCI/GitLab CI:

aduth added a commit that referenced this pull request Jan 6, 2022
**Why**: To simplify ongoing work in #5746, to avoid double-processing of SASS files, and to consolidate in preparation for future CSS bundling refactoring (cssbundling-rails).
aduth added a commit that referenced this pull request Jan 7, 2022
* Standardize SASS processing to Sprockets

**Why**: To simplify ongoing work in #5746, to avoid double-processing of SASS files, and to consolidate in preparation for future CSS bundling refactoring (cssbundling-rails).

* Alphabetize
@aduth aduth force-pushed the aduth-jsbundling-webpack branch from df1a527 to 43179cf Compare January 7, 2022 15:00
aduth added a commit that referenced this pull request Jan 7, 2022
**Why**: As discovered in #5746 (e40b020), if forgery prevention is disabled (e.g. test environment), then there is no CSRF token meta tag on the page. Thus, our assumptions that this would always be present are mistaken. To avoid appending an invalid null token to relevant requests in an environment where forgery protection is disabled, we should properly handle the possibility of its absence.
@aduth aduth changed the title Try: Replace Webpacker with jsbundling-rails + Webpack Replace Webpacker with jsbundling-rails + Webpack Jan 7, 2022
@aduth
Copy link
Member Author

aduth commented Jan 7, 2022

For better compatibility and to anticipate a similar adoption of cssbundling-rails, it may be worth reintroducing it.

Couple interop improvements here:

  • Renamed public/packs/assets-manifest.json back to public/packs/manifest.json, as it was with Webpacker (b88a941)
  • Reintroduced nested public/packs/js directory, as it was with Webpacker (afb80bf)

Also confirming that CloudFront/S3 assets are being uploaded correctly during build:

upload: [REDACTED]/public/packs/js/masked-text-toggle-5bf75821.js to s3://[REDACTED]/packs/js/masked-text-toggle-5bf75821.js

aduth added a commit that referenced this pull request Jan 7, 2022
* Change document capture CSRF handling to optional

**Why**: As discovered in #5746 (e40b020), if forgery prevention is disabled (e.g. test environment), then there is no CSRF token meta tag on the page. Thus, our assumptions that this would always be present are mistaken. To avoid appending an invalid null token to relevant requests in an environment where forgery protection is disabled, we should properly handle the possibility of its absence.

* specs
@aduth aduth force-pushed the aduth-jsbundling-webpack branch from 89303b3 to 761f6d9 Compare January 7, 2022 17:57
@aduth aduth marked this pull request as ready for review January 7, 2022 19:08
aduth added 12 commits January 11, 2022 15:07
More interop with legacy Webpacker. Also, we may be building CSS in the future.
**Why**: Webpacker fell back automatically to RAILS_ENV, which is assigned in Circle configuration. Use an explicit NODE_ENV instead, and restore Webpack missing key handling behavior.
Mode can only be "production" or "development", not an arbitrary NODE_ENV value (e.g. "test")
Needed in build environments
Now outputting to nested js/ directory
By default, Webpack output uses arrow function syntax, obviously not supported in IE. Thanks es5-safe!
This is the default for jsbundling-rails, but we're outputting to public instead
Lots of places we set RAILS_ENV, leading to inconsistencies if we don't also set NODE_ENV (e.g. bin/migrate for devops build)
Would have thought this would be automatic by "mode"!
@aduth aduth force-pushed the aduth-jsbundling-webpack branch from 8af5bb5 to 5160075 Compare January 11, 2022 20:12
@aduth
Copy link
Member Author

aduth commented Jan 11, 2022

Another quality-of-life regression with this is that running feature specs locally will assume that the latest-built JavaScript already exists, whereas previously Webpacker would integrate into the request stack to kick off a compilation if needed. I don't think this is strictly a blocker since it's a matter of running yarn build to resolve, but happy to be overruled.

@mitchellhenke
Copy link
Contributor

Another quality-of-life regression with this is that running feature specs locally will assume that the latest-built JavaScript already exists, whereas previously Webpacker would integrate into the request stack to kick off a compilation if needed. I don't think this is strictly a blocker since it's a matter of running yarn build to resolve, but happy to be overruled.

It would be nice if make test at least compiled?

@aduth
Copy link
Member Author

aduth commented Jan 11, 2022

It would be nice if make test at least compiled?

Yeah, that seems like a reasonable compromise for now 👍 I'll add it.

Edit: See 08bfc38.

@aduth aduth merged commit 0e4fbc7 into main Jan 12, 2022
@aduth aduth deleted the aduth-jsbundling-webpack branch January 12, 2022 13:06
aduth added a commit that referenced this pull request Jan 13, 2022
**Why**: One of the other advantages / hopes with #5746 was a more direct path to incrementally adopting native TypeScript if we choose.
aduth added a commit that referenced this pull request Jan 14, 2022
* Try: TypeScript

**Why**: One of the other advantages / hopes with #5746 was a more direct path to incrementally adopting native TypeScript if we choose.

* Move babel/preset-typescript to dependencies

Required for production build

* wip: ESLint, Mocha

* Move TypeScript ESLint rules to shared configuration

* Revert extends to main

Simplify diff

* Add test case for additional resolved Webpack extensions
@aduth
Copy link
Member Author

aduth commented Jan 31, 2022

It would be nice if make test at least compiled?

Yeah, that seems like a reasonable compromise for now 👍 I'll add it.

Edit: See 08bfc38.

After the fact, I notice that jsbundling-rails will auto-enhance a test:prepare task with the build:

https://github.com/rails/jsbundling-rails/blob/98780f2/lib/tasks/jsbundling/build.rake#L14-L15

I assume that's not automatically invoked, but that someone could run rake test:prepare before running tests? It might be an alternative for what was added in 08bfc38, though at least based how those targets are arranged, I don't feel particularly compelled to change how it's implemented.

@justin808
Copy link

justin808 commented Feb 4, 2022

Hi @aduth @mitchellhenke @zachmargolis,

@tomdracz, and I have been putting a lot of effort into https://github.com/shakacode/shakapacker, the successor to webpacker.

V6 addressed many of the issues of v5 and prior.

In fact, upon a closer look at the PR description, #5746 (comment), I think shakapacker v6 addresses every one of your concerns.

Since you guys seem to like features like HMR and code-splitting, I'm curious why the transition away from webpacker is helpful?

I took a look at the code and the comments, including this follow-up PR https://github.com/18F/identity-idp/pull/5805/files

It seems that you're reinventing a lot of what is already in webpacker v6, and you'd be a lot better off with shakapacker. We could work together to solve these issues that you're seeing.

And we can be progressive. For example, we've even got SWC.rs support started, added in shakacode/shakapacker#29.

FWIW, I've been working on webpack integration since this article I published in 2014, and I'm the creator of React on Rails.

Anyway, I'm reaching out to see if you'd be interested in working with us on Shakapacker.

@aduth
Copy link
Member Author

aduth commented Feb 4, 2022

Hi @justin808, thanks for reaching out.

You're absolutely right to point out that a lot of what we've implemented here is reinventing the wheel. As mentioned in the original comment, we were beginning to feel held back by Webpacker v5 (Webpack v4) and, absent any alternative known to us at the time, this felt like the most viable path forward.

That being said, our requirements are such that we were already hitting the limits of what we could do within Webpacker. In particular, the code that you reference here has largely existed already from when we were previously using Webpacker, originally introduced in #5223 as a way to allow us a sort of "language pack script" functionality that was only possible by reimplementing parts of the Webpacker manifest script lookup behavior.

I'm glad to see Webpacker continue to live on through Shakapacker. However, speaking for myself, I'm quite happy with the result of the work here and the direct control it allows us over the build configuration. If we start to run into problems with our current approach, we’ll definitely take another look at Shakapacker.

@justin808
Copy link

@aduth wrote

we were already hitting the limits of what we could do within Webpacker

Anyway, I'd love to know what Webpacker limits suggest that the extra code you all added to your project is better than Shakapacker v6+.

I think what you wrote applies to older versions of webpacker.

The new v6+ gem:

  1. has no version dependencies
  2. use webpack-merge to allow easy customization
  3. uses a single file webpack.config.js that can be completely custom
  4. users a YAML file to keep the view helper in sync with the webpacker configuration
  5. has zero requirements that you use the JS webpack part. You can use the view helper, which provides the elegant proxying feature (and caching of the manifest) and other goodness.
  6. support for SWC.rs

(@tomdracz, did I miss anything?)

I think you can delete a lot of the boilerplate you added and have a small webpack.config.js that uses webpack-merge and does a few simple customizations of the default and then 💥 you got what you have now and more!

Now that we can quickly iterate on Shakapacker, I think it will be super-amazing super-quickly 🚀 !

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants