#
Develop a module in isolation
To develop their own independent module, a team should not need to install the host application or any other modules of the application they do not own. However, they should have a way to integrate their module with the application shell (e.g., RootLayout
, RootErrorBoundary
, etc..) while working in isolation.
To achieve this, the first step is to extract the application shell from the host application. There are various ways to accomplish this, but in this guide, we'll transform the host application into a monorepo and introduce a new local package named @sample/shell
specifically for this purpose:
monorepo
├── app
├── libs
├────── shell
├───────── src
├─────────── RootLayout.tsx
├─────────── RootErrorBoundary.tsx
├─────────── AppRouter.ts
├─────────── register.tsx
├─────────── index.ts
├───────── package.json
├── modules
├───────── local-module
├───────── remote-module
#
Create a shell package
The implementation details of the RootLayout
, RootErrorBoundary
and ModuleErrorBoundary
components won't be covered by this guide as it already has been covered many times by other guides.
First, create a new package (we'll refer to ours as shell
) and add the following fields to the package.json
file:
{
"name": "@sample/shell",
"version": "0.0.1",
"type": "module",
"exports": "./src/index.ts"
}
Then, install the package dependencies and create an AppRouter
component in the shell package to provide a reusable router configuration that can be shared between the host application and the isolated modules. This new AppRouter
component should wrap the @squide/firefly
AppRouter component:
import { AppRouter as FireflyAppRouter } from "@squide/firefly";
import { RouterProvider, createBrowserRouter } from "react-router-dom";
import { RootErrorBoundary } from "./RootErrorBoundary.tsx";
export function FireflyAppRouter() {
return (
<AppRouter waitForMsw={false}>
{({ rootRoute, registeredRoutes, routerProviderProps }) => {
return (
<RouterProvider
router={createBrowserRouter([
{
element: rootRoute,
errorElement: <RootErrorBoundary />
children: registeredRoutes
}
])}
{...routerProviderProps}
/>
);
}}
</AppRouter>
);
}
Finally, create a local module to register the application shell. This module will be used by both the host application and the isolated modules:
import { PublicRoutes, ProtectedRoutes, type ModuleRegisterFunction, type FireflyRuntime } from "@squide/firefly";
import { RootLayout } from "./RootLayout.tsx";
import { ModuleErrorBoundary } from "./ModuleErrorBoundary.tsx";
export const registerShell: ModuleRegisterFunction<FireflyRuntime> = runtime => {
runtime.registerRoute({
element: <RootLayout />,
children: [
{
errorElement: <ModuleErrorBoundary />,
children: [
PublicRoutes,
ProtectedRoutes
]
}
]
}, {
hoist: true
});
};
This guide only covers the RootLayout
, RootErrorBoundary
and ModuleErrorBoundary
components but the same goes for other shell assets such as an AuthenticationBoundary
component.
#
Update the host application
Now, let's revisit the host application by adding the new @sample/shell
package as a dependency:
{
"dependencies": {
"@sample/shell": "0.0.1"
}
}
Then, integrate the AppRouter
component from the @sample/shell
package into the application:
import { AppRouter } from "@sample/shell";
export function App() {
return (
<AppRouter />
);
}
And finally include the registerShell
function to setup the RootLayout
and RootErrorBoundary
components as well as any other shell assets:
import { createRoot } from "react-dom/client";
import { ConsoleLogger, RuntimeContext, FireflyRuntime, bootstrap, type RemoteDefinition } from "@squide/firefly";
import { App } from "./App.tsx";
import { registerHost } from "./register.tsx";
import { registerShell } from "@sample/shell";
const Remotes: RemoteDefinition[] = [
{ name: "remote1" }
];
const runtime = new FireflyRuntime({
loggers: [x => new ConsoleLogger(x)]
});
await bootstrap(runtime, {
// Register the newly created shell module.
localModules: [registerShell, registerHost],
remotes: Remotes
});
const root = createRoot(document.getElementById("root")!);
root.render(
<RuntimeContext.Provider value={runtime}>
<App />
</RuntimeContext.Provider>
);
#
Setup a remote module
With the new shell
package in place, we can now configure the remote module to be developed in isolation. The goal is to start the module development server and render the module pages with the same layout and functionalities as if it was rendered by the host application.
To begin, let's start by adding a dependency to the @sample/shell
package:
{
"dependencies": {
"@sample/shell": "0.0.1"
}
}
Then, create the following files in the remote module application:
remote-module
├── public
├──── index.html
├── src
├────── dev
├────────── DevHome.tsx
├────────── register.tsx
├────── register.tsx
├────── Page.tsx
├────── index.tsx
├────── App.tsx
├── webpack.dev.js
├── package.json
#
index.tsx
The index.tsx
file is similar to the bootstrap.tsx
file of an host application but, tailored for an isolated module. The key distinctions are that all the modules are registered as local modules, and a new registerDev
function is introduced to register the development homepage (which will be covered in an upcoming section):
import { createRoot } from "react-dom/client";
import { ConsoleLogger, RuntimeContext, FireflyRuntime, bootstrap } from "@squide/firefly";
import { App } from "./App.tsx";
import { register as registerModule } from "./register.tsx";
import { registerDev } from "./dev/register.tsx";
import { registerShell } from "@sample/shell";
// Loggers, etc... could be reuse through a
// shared packages or faked when in isolation.
const runtime = new FireflyRuntime({
loggers: [x => new ConsoleLogger(x)]
});
await bootstrap(runtime, {
// Registering the remote module as a local module because the "register" function
// is local when developing in isolation.
localModules: [registerModule, registerDev, registerShell]
});
const root = createRoot(document.getElementById("root")!);
root.render(
<RuntimeContext.Provider value={runtime}>
<App />
</RuntimeContext.Provider>
);
#
App.tsx
The App.tsx
file uses the newly created AppRouter
component to setup Squide's primitives with a React Router instance:
import { AppRouter } from "@sample/shell";
export function App() {
return (
<AppRouter />
);
}
#
DevHome.tsx
The DevHome
component is the homepage when the remote module is developed in isolation:
function DevHome() {
return (
<div>
<h2>Remote module development home page</h2>
<p>Hey!</p>
</div>
);
}
To register the development homepage, create a new local module specifically for configuring the remote during isolated development:
import type { ModuleRegisterFunction, FireflyRuntime } from "@squide/firefly";
import { DevHome } from "./DevHome.tsx";
export const registerDev: ModuleRegisterFunction<FireflyRuntime> = runtime => {
runtime.registerRoute({
index: true,
element: <DevHome />
});
}
#
Add a new CLI script
Next, add a new dev-isolated
script to the package.json
file to start the local development server in isolation:
{
"dev": "webpack serve --config webpack.dev.js",
"dev-isolated": "cross-env ISOLATED=true webpack serve --config webpack.dev.js",
}
If your project's package.json
file does not already include the cross-env dependency, be sure to install cross-env
as a development dependency.
The dev-isolated
script is similar to the dev
script but introduces an ISOLATED
environment variable. This variable will be used by the webpack.dev.js
file to conditionally configure the development server to either serve the module as an application for isolated development or as a remote endpoint by the host application through the /remoteEntry.js
entry point.
#
Configure webpack
First, open the public/index.html
file created at the beginning of this guide and copy/paste the following HtmlWebpackPlugin template:
<!DOCTYPE html>
<html>
<head>
</head>
<body>
<div id="root"></div>
</body>
</html>
Then, open the .browserslist
file and copy/paste the following content:
extends @workleap/browserslist-config
#
Isolated environment configuration
To configure webpack, open the webpack.dev.js
file and update the configuration to incorporate the ISOLATED
environment variable and the defineDevHostConfig function:
// @ts-check
import { defineDevRemoteModuleConfig, defineDevHostConfig } from "@squide/firefly-webpack-configs";
import { swcConfig } from "./swc.dev.js";
let config;
if (!process.env.ISOLATED) {
config = defineDevRemoteModuleConfig(swcConfig, "remote1", 8081);
} else {
config = defineDevHostConfig(swcConfig, "remote1", 8080, []);
}
export default config;
If you encounter issues configuring webpack, refer to the @workleap/webpack-configs documentation.
#
Try it 🚀
Start the remote module in isolation by running the dev-isolated
script. The application shell should wrap the pages of the module and the default page should be DevHome
.
#
Troubleshoot issues
If you are experiencing issues with this section of the guide:
- Open the DevTools console. You'll find a log entry for each registration that occurs and error messages if something went wrong.
- Refer to a working example on GitHub.
- Refer to the troubleshooting page.
#
Setup a local module
Similarly to remote modules, local modules can also be set up for isolated development. The key difference is that the webpack.config.js
file for a local module strictly serves the purpose of starting a development server for isolated development. Typically, local modules do not depend on webpack or Module Federation.
First, open a terminal at the root of the local module project and install the @squide/firefly-webpack-configs
package and its dependencies:
pnpm add -D @workleap/webpack-configs @workleap/swc-configs @workleap/browserslist-config @squide/firefly-webpack-configs webpack webpack-dev-server webpack-cli @swc/core @swc/helpers browserslist postcss
yarn add -D @workleap/webpack-configs @workleap/swc-configs @workleap/browserslist-config @squide/firefly-webpack-configs webpack webpack-dev-server webpack-cli @swc/core @swc/helpers browserslist postcss
npm install -D @workleap/webpack-configs @workleap/swc-configs @workleap/browserslist-config @squide/firefly-webpack-configs webpack webpack-dev-server webpack-cli @swc/core @swc/helpers browserslist postcss
While you can use any package manager to develop an application with Squide, it is highly recommended that you use PNPM as the guides has been developed and tested with PNPM.
Then, add a peer dependency and a dev dependency to the @sample/shell
package:
{
"peerDependencies": {
"@sample/shell": "*"
},
"devDependencies": {
"@sample/shell": "0.0.1"
}
}
Then, create the following files in the local module application:
local-module
├── public
├────── index.html
├── src
├────── dev
├────────── DevHome.tsx
├────────── register.tsx
├────── register.tsx
├────── Page.tsx
├────── index.tsx
├────── App.tsx
├── .browserslistrc
├── swc.config.js
├── webpack.config.js
├── package.json
#
index.tsx
This file is similar to the index.tsx
file of the
#
App.tsx
This file is similar to the App.tsx
file of the
#
DevHome.tsx and registerDev
These files are similar to the dev/DevHome.tsx
and dev/register.tsx
files of the
#
Configure webpack
First, open the public/index.html
file and copy/paste the following HtmlWebpackPlugin template:
<!DOCTYPE html>
<html>
<head>
</head>
<body>
<div id="root"></div>
</body>
</html>
Then, open the .browserslist
file and copy/paste the following content:
extends @workleap/browserslist-config
Then, open the swc.config.js
file and copy/paste the following code:
// @ts-check
import { browserslistToSwc, defineDevConfig } from "@workleap/swc-configs";
const targets = browserslistToSwc();
export const swcConfig = defineDevConfig(targets);
Finally, open the webpack.config.js
file and use the the defineDevHostConfig function to configure webpack:
// @ts-check
import { defineDevHostConfig } from "@squide/firefly-webpack-configs";
import { swcConfig } from "./swc.config.js";
export default defineDevHostConfig(swcConfig, "local1", 8080, []);
If you encounter issues configuring webpack, refer to the @workleap/webpack-configs documentation.
#
Add a new CLI script
Next, add a new dev-isolated
script to the package.json
file to start the local development server:
{
"dev-isolated": "webpack serve --config webpack.config.js"
}
#
Try it 🚀
Start the remote module in isolation by running the dev-isolated
script. The application shell should wrap the pages of the module and the default page should be DevHome
.
#
Troubleshoot issues
If you are experiencing issues with this section of the guide:
- Open the DevTools console. You'll find a log entry for each registration that occurs and error messages if something went wrong.
- Refer to a working example on GitHub.
- Refer to the troubleshooting page.