background

Imagine A scenario where team A develops A component library that both team B and team C use in their respective business projects. Now team A needs to update A component (for example, change the color). Traditionally, team A would release A new version, and then the other two teams would update each version of the component library that the business project relies on and release it.

Is there a faster way? For example, if only the component library can be updated, other projects that depend on it can automatically obtain its latest version, which implements remote dynamic components. The new Module Federation in Webpack 5 can do this, but today we’ll talk about a different approach.

Remote dynamic component implementation

Remote dynamic component library

The remote Dynamic component library project structure is as follows:

.├ ── ├─ ├─ ├─ ├── ├─ ├── ├── ├.js ├── ├.js ├── ├.js ├── ├.js ├── ├.js ├── ├.js ├── ├.js ├── ├.jsCopy the code

We simply implement two components: Button and Text:

import React from 'react'

export default ({children}) => {
  return <button style={{color: 'blue'}} >{children}</button>
}
Copy the code
import React from 'react'

export default ({children}) => {
  return <span style={{color: 'blue'}} >{children}</span>
}
Copy the code

We use rollup to package it. We use rollup because the packaged code is very concise and easy to view. Rollup is configured as:

import babel from 'rollup-plugin-babel'
import fs from 'fs'

const components = fs.readdirSync('./src')

export default components.map((filename) = > {
  return {
    input: `./src/${filename}`.output: {
      file: `dist/${filename}`.format: 'cjs',},plugins: [babel()],
  }
})
Copy the code

The result is as follows:

.├ ── │ ├─ ├─ ├.jsCopy the code

Button.js is as follows:

'use strict'

var React = require('react')

function _interopDefaultLegacy(e) {
  return e && typeof e === 'object' && 'default' in e ? e : {default: e}
}

var React__default = /*#__PURE__*/ _interopDefaultLegacy(React)

var Button = function (_ref) {
  var children = _ref.children
  return /*#__PURE__*/ React__default['default'].createElement(
    'button',
    {
      style: {
        color: 'blue',
      },
    },
    children
  )
}

module.exports = Button
Copy the code

– and then we use HTTP server on a static file in the dist directory services, can be obtained via http://localhost:8080/Button.js to code after packaging.

With the introduction of the remote component library, you’ll see how to use it in business projects.

Access the remote component library

Create a React app with create-react-app and add a DynamicComponent:

const DynamicComponent = ({name, children, ... props}) = > {
  const Component = useMemo(() = > {
    return React.lazy(async () => fetchComponent(name))
  }, [name])

  return (
    <Suspense
      fallback={
        <div style={{alignItems: 'center', justifyContent: 'center', flex: 1}} >
          <span style={{fontSize: 50}} >Loading...</span>
        </div>} ><Component {. props} >{children}</Component>
    </Suspense>)}export default React.memo(DynamicComponent)
Copy the code

Suspense and React. Lazy methods are used in Suspense and React. The DynamicComponent loads the target component remotely. The component renders what is passed in Suspense parameter fallback and is eventually replaced with the component that has successfully loaded. Let’s see how fetchComponent is implemented:

const fetchComponent = async (name) => {
  const text = await fetch(
    ` http://127.0.0.1:8080/${name}.js? ts=The ${Date.now()}`
  ).then((a) = > {
    if(! a.ok) {throw new Error('Network response was not ok')}return a.text()
  })
  const module = getParsedModule(text, name)
  return {default: module.exports}
}
Copy the code

This method makes a network request to get the component’s code, which gets parsed by getParsedModule and returned by the module. Let’s look at how getParsedModule is implemented:

const packages = {
  react: React,
}

const getParsedModule = (code) = > {
  let module = {
    exports: {},}const require = (name) = > {
    return packages[name]
  }
  Function('require, exports, module', code)(require.module.exports, module)
  return module
}
Copy the code

Function is used to run the incoming code, since the react library is not packaged with the remote component, so require is implemented.

Let’s look at this code with the button.js package, which is actually equivalent to the following code:

const packages = {
  react: React,
}

const getParsedModule = (code, moduleName) = > {
  let module = {
    exports: {},}const require = (name) = > {
    returnpackages[name] } ; ((require.exports.module) = > {
    'use strict'

    var React = require('react')

    function _interopDefaultLegacy(e) {
      return e && typeof e === 'object' && 'default' in e ? e : {default: e}
    }

    var React__default = /*#__PURE__*/ _interopDefaultLegacy(React)

    var Button = function (_ref) {
      var children = _ref.children
      return /*#__PURE__*/ React__default['default'].createElement(
        'button',
        {
          style: {
            color: 'blue',
          },
        },
        children
      )
    }

    module.exports = Button
  })(require.module.exports, module)
  return module
}
Copy the code

Finally, we can use the DynamicComponent as follows:

import DynamicComponent from './DynamicComponent'

function App() {
  return (
    <div>
      <DynamicComponent name='Button'>Click Me</DynamicComponent>
      <DynamicComponent name='Text'>Remote Component</DynamicComponent>
    </div>)}export default App
Copy the code

Now let’s try to modify the components in the remote dynamic component library and repackage them to see the results immediately.

conclusion

This article describes a way to implement remote dynamic components, but it is rudimentary, and in fact we have a lot of room for optimization. For example, the way the page is implemented today, if you use two buttons on top of the page, it will make two requests, which is obviously not reasonable. We can solve this problem by pre-loading and module caching.