Federated Modules Production Grade boilerplate
To run this project the following package versions are required:
- NPM v8.x.x
- Node v16.14.1 (configured via .nvmrc)
- Run
npm i
at the root of the Monorepo to install the required node modules and symlink all the libraries (they will also need to build). - Run
npm run build-all
to build all of the applications and libraries
To run the fully intergrated application in development mode you need to run the App Shell along with the required applications:
// Terminal 1
npm run app-shell:dev
// Terminal 2
npm run footer:dev
// Terminal 3
npm run header:dev
Once the servers have successfully compiled visit http://localhost:3000/app
to access the
App Shell.
Each federated micro-frontend exposes a /prerender
endpoint for parent components to consume, and is available
standalone at /app
, e.g. the footer can be viewed (along with any micro-frontends it consumes) at
http://localhost:3003/app
.
To run the application in production mode (with Hot Module Reloading) replace the dev
command with watch
:
// Terminal 1
npm run app-shell:watch
Testing can be done from the root via commands like npm run host:test
Testing can be done from the root via commands like npm run host:eslint
Build tsdocs with npm run docs
and serve docs locally with npm run docs:serve
npm 8 workspaces have a number of features which make working with
large monorepo projects simpler. Workspaces are defined in the root package's package.json
, e.g.
"workspaces": [
"./cool-package-1",
"./cool-package-2",
],
We can then run scripts from those packages without having to change directory, e.g.
$ npm run build -w cool-package-1
or run the same script in all workspaces
$ npm run build --ws
Note the -w
argument refers to the name defined in the workspace's package.json
, not the path used above.
When installing dependencies, by default they will be installed into the root package's node_modules
and symlinked
into the child packages. This speeds up installation and reduces the amount of disk space required when using the same
dependency across different packages.
Functionality can be shared across packages without the overhead of publishing to a registry like npm, e.g.
// cool-package-1/index.js
import coolFunction from 'cool-package-2'
- Each MFE has its own Redux state, and can render standalone at
http://localhost:[port]/app
- When the module is federated into the app-shell, that state is read from the app-shell's top-level store
- initial state is passed from the MFE to the app-shell via prerender middleware and the
federate-component
package, and collated together when the app-shell client bootstraps itself - each MFE exports a reducer via client-side module federation, which get combined with the app-shell reducer before the app hydrates on the client
- this way, although the MFEs share a common store, they can't (or at least shouldn't) affect the state of other MFEs.
- Add redux to standalone app using redux toolkit, build state, reducers etc and apply to components using
connect
- export the reducer as a default export
- add the file that exports the reducer to the MFE's
exposes
config, e.g.exposes: {'./store': './src/store.ts',}
- in the consumer, update the REMOTE_URLS env variable to use the format
"mfe_name":"mfe_name@mfe_url"
- in the consumer's client bootstrap file, import the reducer from the remote
import mfe_reducer from 'mfe_name/store
- when initialising redux on the client, combine the reducers from all the remotes with the initial state that gets passed through the initial state middleware