This is an internal presentation for September 2021, and the PPT download is at the end of the presentation. It will be released in February 2022. Article content has absorbed the essence of many community articles, combined with the production of the ground to summarize, if there is infringement, please contact delete. Because the original part involves internal data, it has been trimmed, which may lead to unclear description.

Technical points: Qiankun, Vue3, Vue Cli, Element Plus

Abstract

The front end, the tech world, the entertainment world. Under the trend of big front-end, the previous front-end development mode can no longer bear the actual project needs, we need a series of schemes to make our projects standardized, configurable, optimized, etc. The big thing this year is the micro front end and low code. Low code is not involved. Let’s talk about the micro front end. Tell me about what’s going on.

background

Due to the project size (3 years of development, 5 years of promotion, X billion investment) and planning and product requirements, support independent launching and updating of functional modules to reduce the risk of single launching. Participate in 4+ development teams (50+ for the largest number of team members in different locations).

After many times of communication, the following conclusions and corresponding problems were drawn according to the requirements:

  • The platform system has large volume and many functions

    • Low development efficiency
    • Multi-person collaboration is costly
    • High access cost
  • Platform systems are long and sensitive

    • Active period length
    • High availability requirements
    • Redundant construction may not be managed

Appeal:

Build systems that are well experienced and sustainably maintained

The core solves two scenarios:

  • Based on product (experience) latitude
  • Latitude based on technology architecture

For these reasons, the decision was made to adopt a micro front end solution.

Stonehenge applications, IFrame, framework components, and microfronts can all solve these problems. Each of the four approaches has advantages and disadvantages. The core of the micro front-end solution is to find a balance between experience and efficiency. Based on the investigation and comparison, we finally decided to choose the technical solution of micro front end to upgrade the business architecture.

Microfront-end concept

The micro-front-end architecture is similar to the micro-service architecture. It applies the concept of micro-service to the browser side, that is, the Web application is transformed from a single single application to multiple small front-end applications.

The change is that these front-end applications can be independently run, independently developed, and independently deployed. Also, they should be able to be developed in parallel while sharing components — components that can be managed through NPM or Git tags, Git submodules, and so on.

A microfront-end is an architectural style that combines a number of independently delivered front-end applications into a large whole.

Of course, there is no free lunch in software architecture: everything comes at a cost. Some micro front-end implementations can lead to repeated dependencies, forcing users to download more content. In addition, the increased level of team autonomy can lead to more fragmentation of teams’ work. However, we believe that these risks can be controlled at a reasonable level, and that the benefits of the micro front end ultimately outweigh the disadvantages.

Read more about the microfront end in detail

When do you use a micro front end

  • Large-scale enterprise Web application development
  • Collaborative development across teams and enterprise applications
  • Long-term returns are higher than short-term ones
  • Different technology selection projects
  • Part of a cohesive single product needs independent publishing, grayscale and other capabilities

Vernacular is suitable for old projects, boulder applications, projects with many collaborators.

The micro front end is hot, but don’t treat it like a silver bullet. It’s good for old projects, boulder applications, and projects with lots of collaborators. If there is no special requirement, the simplest way to do it is the conventional solution.

Micro front-end solutions

  • Single-spa is described on its website as a meta-framework that allows you to integrate multiple different frameworks on a single page. Many microfront end solutions are based on this secondary development or inspiration to support ESM
  • qiankunBased on thesingle-spaMicro front-end solutions are available for production
  • Icestark is a micro front end solution for large applications
  • MicroApp A minimalist solution for building microfront-end applications with ESM support (need to turn off the sandbox)
  • Garfish contains the basic capabilities needed to build microfront-end systems that can be used by any front-end framework. Simple access, can easily combine multiple front-end applications into a cohesive single product. Sandbox isolation mechanism is improved
  • Emp is a micro front-end solution built based on Webpack5 Module Federation

As the project started at the beginning of 2021, the solution chosen was Qiankun based on the open source situation at that time. If you were to start over now, support for ES Modules, sandbox implementation mechanisms, and isolation levels would also be the core reference points of choice.

If there is no special requirement, it is recommended to use the scheme with the largest number of users.

If sandbox is required, try Garfish;

If you don’t have the requirement of background label switching to save state, you can try MicroApp.

If a micro-application is developed and deployed by a third party and cannot require cross-domain Settings, icestARK is recommended.

Go out of your way to use Single-SPA and support everything.

Technology stack

  • Core: Vue3, Vue CLI 5, TypeScript
  • UI library: Element Plus
  • Unit test: Jest
  • E2E: TestCafe

Implementation process

Architecture diagram

The framework application is responsible for the overall Layout and micro-application configuration and registration rendering. As you can see from the above figure, the framework application has a general Header and a siderBar. In addition to Layout, you also need to configure the micro-application information, such as bundle URL, baseline routing, etc. Microapplications are applications that are broken down by business dimension, usually a SPA application, and contain at least one to more pages or routes.

In principle, framework applications should avoid UI code that contains specific pages. If a framework application does too many things, it may cause the following problems:

  • Too much framework application style code increases the probability of microapplication and framework application style conflicts

  • Framework applications provide microapplications with other capabilities, such as global apis, that break microapplications’ independence and increase their coupling

  • The essence of framework application is a centralized part. After the change, it is necessary to return to all micro-applications in principle, so it is necessary to ensure the simplicity of responsibility. The simpler things are, the more stable they are

The flow chart

There are two ways to look at the workflow of incorporating the micro front-end architecture. One is the development pattern of the micro application on the right. Microapplication development has independent warehouse, independent development, testing, deployment process. After the development test is deployed, the release artifacts of the application are registered with the framework application. These artifacts may be JS bundles or HTML resources.

On the left is the overall flow of a framework application, which maintains the micro-application registration information. When a user accesses the system, it can accurately match the application information to be loaded according to the route information registered before, load application resources according to the corresponding information, and finally render the application.

When a user clicks to trigger a redirect, if the route change triggers an internal app redirect, the app will render the page based on the internal app routing logic. If some cross-application hops are involved, it’s back to the route lookup process above.

Due to the design requirements of the deployment architecture, the project will adopt Multirepo. If conditions support, I think it is better to use Monorepo

Applications access

The use of Qiankun does not have a high degree of intrusion into the project, which is not much different from the conventional SPA development. Adjustment points are as follows:

The main application

  1. Add sub-applications and rules

    // src/micro/app.ts
    const apps: RegistrableApp<IAppProps>[] = [{
      name: 'emd-app'./ / https://next.cli.vuejs.org/zh/guide/mode-and-env.html#%E7%8E%AF%E5%A2%83%E5%8F%98%E9%87%8F environment variables
      // If the registration logic is dynamically configured to deliver, the application orchestration can be implemented
      entry: process.env.VUE_APP_EMD_APP_URL as string.container: '#frame'.activeRule: '/emd'}, {name: 'et-app'.entry: process.env.VUE_APP_ET_APP_URL as string.container: '#frame'.activeRule: '/et'},... ] ;export default apps;
    Copy the code
    // src/micro/index.ts
    import {
      registerMicroApps, start, addGlobalUncaughtErrorHandler, runAfterFirstMounted, setDefaultMountApp
    } from 'qiankun';
    import NProgress from 'nprogress';
    import 'nprogress/nprogress.css';
    import apps from './apps';
    
    /** * Register child application * first argument - registration information for child application * second argument - global lifecycle hook */
    registerMicroApps(apps, {
      / / before loading
      beforeLoad: () = > {
        NProgress.start();
        return Promise.resolve();
      },
      / / after a mount
      afterMount: () = > {
        NProgress.done();
        return Promise.resolve(); }});/** * Adds a global uncaught exception handler */
    addGlobalUncaughtErrorHandler((event: Event | string) = > {
      const { message: msg, error } = event as any;
      // Load failed
      if (msg && msg.includes('died in status LOADING_SOURCE_CODE')) {
        console.error(` child application${error? .appOrParcelName}Loading failed. Please check whether the application is running); }});/** * sets the default subapplication */
    setDefaultMountApp('/');
    
    /** * runAfterFirstMounted */
    runAfterFirstMounted(() = > {
      // console.log('[MainApp] first app mounted');
    });
    
    // Export the startup function of Qiankun
    export default start;
    
    Copy the code
  2. Add subapplication render routes

    {
      path: ':micro(emd|et|rpt):endPath(.*)'.name: 'MicroApp'.component: () = > import('@/views/MicroApp.vue')}Copy the code
  3. Implement sub-application loading

    // @/views/MicroApp.vue
    <template>
      <div id="frame"></div>
    </template>
    
    <script setup lang='ts'>
    import { onMounted } from 'vue';
    import start from '@/micro';
    
    onMounted(() = > {
      if (window.qiankunStarted) return;
      window.qiankunStarted = true;
      start();
    });
    </script>
    Copy the code

    If the window variable warning appears, you can add it in shims-vue.d.ts

    // shims-vue.d.ts
    interface Window {
      qiankunStarted: boolean
    }
    Copy the code

The child application

// main.ts
let emdRouter = null;
let emdApp: any = null;
let emdHistory: any = null;

function render(props: any = {}) {
  const { container } = props;
  // A secondary domain name needs to be added for security purposes
  const urlPrefix = `${process.env.VUE_APP_URL_PRE || ' '}/emd`;
  // eslint-disable-next-line no-underscore-dangle
  emdHistory = createWebHistory(window.__POWERED_BY_QIANKUN__ ? urlPrefix : '/');
  emdRouter = createRouter({
    history: emdHistory,
    routes
  });

  emdApp = createApp(App);
  emdApp.use(emdRouter);
  emdApp.use(store);
  emdApp.use(i18n);
  emdApp.use(permission);
  emdApp.mount(container ? container.querySelector('#app') : '#app');
}

// eslint-disable-next-line no-underscore-dangle
if (!window.__POWERED_BY_QIANKUN__) {
  render();
}

export async function bootstrap() :Promise<void> {
  console.log('emd-app bootstraped');
}

/** * The application calls the mount method every time it enters, and usually we trigger the application's render method here */
export async function mount(props: any) {
  if (props) {
    // Inject actions instances for application communication
    actions.setActions(props);
  }
  render(props);
}

export async function unmount() {
  emdApp.unmount();
  // eslint-disable-next-line no-underscore-dangle
  emdApp._container.innerHTML = ' ';
  emdApp = null;
  emdRouter = null;
  emdHistory.destroy();
}
Copy the code

The subapplication is connected to the framework application. As long as the subapplication implements the bootstrap, mount, and unmount lifecycle hooks, the framework application knows how to load the subapplication.

When the child application is mounted for the first time, bootstrap is performed for some initialization, and then mount is performed to mount it. If you are a child of the Vue stack, you might want to mount createApp().render, mount your Vue App to a real node, and render the App. When your app switches away, unmount it and it comes back (typical scenario: You jump from app A to app B, and then back to App A.) We don’t need to re-execute all the lifecycle hooks at this point. We don’t need to start with bootstrap.

More details of qiankun access can be found at:

  • Best Practices in Microfront-end based on Qiankun (Swastika) – from 0 to 1
  • Best Practices of Microfront-end based on Qiankun (illustrated) – Inter-application Communication

Nginx deployment

Deploy multiple modules based on the deployment environment requirements. Due to the microapplication architecture, sub-applications need to support cross-domain, cross-domain configuration:

location / {  
    add_header Access-Control-Allow-Origin *;
    add_header Access-Control-Allow-Methods 'GET, POST, OPTIONS';
    add_header Access-Control-Allow-Headers 'DNT,X-Mx-ReqToken,Keep-Alive,User-Agent,X-Requested-With,If-Modified-Since,Cache-Control,Content-Type,Authorization';
}
Copy the code

The potholes I came across

The cache problem

After the main application is updated, the user will occasionally access the old version of the situation. After investigation, it is found that the user gets the old version of the cache index.html; After the update, the child application still accesses the old version file, which is the same as the main application. This is a common problem of SPA. The default cache policy is negotiation cache. You can cancel the cache of corresponding files.

Nginx configuration is as follows:

location = /index.html {
  add_header Cache-Control no-cache;
}
Copy the code

Can also be changed to

if (\$request_filename ~ .*\.(htm|html)$){add_header Cache-Control no-store; }Copy the code

Nginx Zip support

If static resources are not compressed, performance and experience are poor. Zip enable Nginx configuration as follows:

gzip  on;
gzip_min_length  1k;
gzip_buffers     4 16k;
gzip_http_version 1.1;
gzip_comp_level 9;
gzip_types text/plain text/css application/xml text/javascript application/javascript application/json image/svg+xml font/ttf;
gzip_vary on;
Copy the code

Not all files are compressed well. Too small files become larger after compression. Set a compression threshold, for example, 1K

Element-plus package Icon garble

In the test environment, garbled font ICONS may appear. After refreshing, the characters return to normal. Dart – SASS is a Unicode encoding bug.

There are two solutions:

  1. willdart-sassChange fornode-sass
  2. indart-sassProcess before compilingunicodecoding

Solution 1 node-Sass technology is outdated, official maintenance is not available, compilation is slow, domestic download takes a long time and there is a high probability of download failure, which is not considered. That leaves option 2, which uses a PostCSS plugin as prompted by the issue on Dart-Sass

Solution:

  1. Install postcss sass — unicode

  2. Add postcss. Config. Js

    // postcss.config.js
    module.exports = {
      plugins: [
        require('postcss-sass-unicode'),
        require('autoprefixer')]};Copy the code

Add and pack effect

However, there is an interesting problem, that is, the icon review style of Element and Element Plus official website is garbled, but the display is correct, I wonder whether special processing has been added. But it doesn’t matter, the ICONS in the project have now been moved to the SVG icon and Element Plus has been moved. Compared with icon font, although some scenes are inconvenient to use, the final product is much smaller, about several hundred K. Font Icon No matter how many font files you use, the font files must be downloaded in full. SVG Icon can be applied and built on demand.

Element-plus applies on demand

Previously component library on-demand applications needed to be imported on demand within the used components, or globally. The former is inconvenient and needs to be introduced every time, while the latter easily becomes a full introduction. Is there a way to support what components I use to automatically import on demand during the packaging phase? In a share, grandpa shared a magic artifact, just use it, the plugin will help you to use the components in need. Almost all popular Vue UI component libraries are supported, such as Ant Design Vue, Element Plus, Vant, Naive UI

The configuration is as follows:

  1. Install unplugin-vue-components: NPM I unplugin-vue-components -d

  2. Adding a Configuration Plugin

    const Components = require('unplugin-vue-components/webpack');
    const { ElementPlusResolver } = require('unplugin-vue-components/resolvers');
    
    module.exports = {
      chainWebpack: (config) = > {
        config.plugin('Components')
          .use(Components({
            resolvers: [ElementPlusResolver({ importStyle: false})]})); }};Copy the code

KeepAlive support

The product wants to support breadcrumb label switching and cache page state. According to our implementation above, auto-loading child applications does not implement state caching through KeepAlive tags,

After looking at the API description of Qiankun, it was found that the sub-application implementation could be manually loaded using loadMicroApp.

Design ideas

  1. Implement the route mapping record that needs to be cached by applying the navigational guard router. BeforeEach of the principal

  2. Through the communication mechanism of Qiankun, cached information was sent to sub-applications, which were responsible for their own KeepAlive

    The KeepAlive of the micro front-end is different from that of the single. Multiple applications have multiple Vue instances, so each application needs to implement KeepAlive.

  3. According to the routing matching rules, the sub-application is loaded manually and its status is cached

implementation

The main application

Navigational guards, they usually have access intercepts in them.

/** * Whether the cache data exists *@param cachedViews
 * @param view* /
const hasView = (cachedViews: ICachedView[], view: ICachedView) = > cachedViews.some((v) = > v.fullPath === view.fullPath);

/** * Route cache information encapsulation *@param route* /
const cachedViewEncapsulation = (route: RouteLocationNormalized): ICachedView= > ({
  path: route.path,
  fullPath: route.fullPath,
  query: route.query,
  name: route.name || ' '
});

/** * Navigation guard */
router.beforeEach(async (to, from, next) => {
  ........
  // Login and other pages do not need KeepAlive
  if(! to.meta.noKeepAlive) { store.dispatch('addCachedViews', cachedViewEncapsulation(to));
  }
  next();
});
Copy the code

Global state, add cache mapping record. Breadcrumb label name, URL information, etc

export default createStore<IStoreType>({
  state: {
    cachedViews: []},getters: {
    cachedViews: (state): ICachedView[] => state.cachedViews
  },
  mutations: {
    /** * page cache changes *@param state
     * @param view
     * @constructor* /
    ADD_CACHE_VIEWS(state, view) {
      / / fullPath matching
      if (hasView(state.cachedViews, view)) {
        return;
      }
      // Same path multi-label processing, ET module
      if(view? .query? .sign) { state.cachedViews.push({ ... view }); }else {
        // The path is matched and route parameters are updated
        const index = state.cachedViews.findIndex((v) = > v.path === view.path);
        if (index > -1) {
          state.cachedViews[index] = view;
        } else{ state.cachedViews.push({ ... view }); } } actions.setCachedViews(state.cachedViews); },DEL_CACHED_VIEWS(state, view) {
      const index = state.cachedViews.findIndex((v) = > v.fullPath === view.fullPath);
      state.cachedViews.splice(index, 1);
      actions.setCachedViews(state.cachedViews);
    },
    DEL_ALL_CACHED_VIEWS(state) {
      state.cachedViews = [];
      actions.setCachedViews([]);
    },
    DEL_OTHER_CACHED_VIEWS(state, view) {
      if (state.cachedViews.length === 1) return; state.cachedViews = [view]; actions.setCachedViews(state.cachedViews); }},actions: {
    addCachedViews({ commit }, view) {
      commit('ADD_CACHE_VIEWS', view);
    },
    delCachedViews({ commit }, view) {
      commit('DEL_CACHED_VIEWS', view);
    },
    delAllCachedViews({ commit }) {
      commit('DEL_ALL_CACHED_VIEWS');
    },
    delOtherCachedViews({ commit }, view) {
      commit('DEL_OTHER_CACHED_VIEWS', view); }}});Copy the code

Layout of the components

// Layout.vue
<template>
  <el-main>
    <router-view v-slot="{ Component, route }">
      <transition>
        <keep-alive :include="cachedViews">
          <component :is="Component" :key="route.meta.usePathKey ? route.path : undefined"/>
        </keep-alive>
      </transition>
    </router-view>
  </el-main>
</template>

<script setup lang='ts'>
import { computed } from 'vue';
import { useStore } from 'vuex';

const store = useStore();

const cachedViews = computed(() = > store.state.cachedViews.map((v) = > v.name));
</script>
Copy the code

The child application loads the component

// @/views/MicroApp.vue
<template>
  <div>
    <div id="frame"></div>
  </div>
</template>

<script lang='ts'>
import { defineComponent, reactive, watch, onMounted, onUnmounted } from 'vue';
import { useRoute } from 'vue-router';
import { useStore } from 'vuex';
import { loadMicroApp } from 'qiankun';
import NProgress from 'nprogress';
import type { ICachedView } from '@xx/base-core';
import apps from '@/micro/apps';

export default defineComponent({
  name: 'MicroApp'.setup() {
    const microList = reactive<any>({});
    const appRoute = useRoute();
    const appStore = useStore();

    /** * listen for route changes, add/modify/delete cache *@param path* /
    const activationHandleChange = async (path: string) => {
      const activeRules: string[] = apps.map((app) = > app.activeRule as unknown as string);
      const isMicro = activeRules.some((rule) = > path.startsWith(rule));
      if(! isMicro)return;
      const conf = apps.find((app) = > path.startsWith(app.activeRule.toString()));
      if(! conf)return;
      // If it has been loaded once, there is no need to load it again
      const current = microList[conf.activeRule.toString()];
      if (current) return;

      // Cache the current child application
      NProgress.start();
      constmicro = loadMicroApp({ ... conf }); microList[conf.activeRule.toString()] = micro;try {
        await micro.mountPromise;
      } catch (e) {
        console.error(e);
      } finally{ NProgress.done(); }};const hasCachedViews = (key: string, arr: string[]) = > arr.some((url: string) = > url.startsWith(key));
    
    /** * Close the TAB TAB to uninstall the closed child application *@param newVal
     * @param oldVal* /
    const unmountMicApp = (newVal: number, oldVal: number) = > {
      if (newVal > oldVal) return;
      const cachedViewsAppUrls = appStore.state.cachedViews.map((item: ICachedView) = > item.path);
      const keys = Object.keys(microList);
      keys.forEach((key: string) = > {
        if(! hasCachedViews(key, cachedViewsAppUrls)) { microList[key].unmount();deletemicroList[key]; }}); }; watch(() = > appRoute.path, activationHandleChange);

    watch(() = > appStore.state.cachedViews.length, unmountMicApp);

    onMounted(() = > {
      if (window.qiankunStarted) return;
      window.qiankunStarted = true;
      activationHandleChange(appRoute.path);
    });

    onUnmounted(() = > {
      window.qiankunStarted = false;
      Object.values(microList).forEach((mic: any) = >{ mic.unmount(); }); }); }});</script>
Copy the code

If the following exceptions occur, different subapplications can be mounted to different nodes. After a subapplication label is closed, uninstall the subapplication.

The child application

The KeepAlive matching needs to be implemented in app. vue, and the information is obtained from the communication of the Qiankun application

// App.vue
<template>
  <router-view v-slot="{ Component, route }">
    <keep-alive :include="canKeepAlive">
      <component :is="Component" :key="route.fullPath"/>
    </keep-alive>
  </router-view>
</template>

<script setup lang="ts">
import { computed, ref } from 'vue';
import { useStore } from 'vuex';
import { actions } from '@xx/base-core';
import type { ICachedView } from '@xx/base-core';
import routes from '@/router';

const store = useStore();

const cacheViews = ref<ICachedView[]>([]);

actions.onGlobalStateChange((state: Record<string, any>) = > {
  cacheViews.value = state.cachedViews;
}, true);

/** * Cache component name array */
const canKeepAlive = computed(() = > {
  // emd is the prefix of the child application
  const emdCacheViews = cacheViews.value.filter((item) = > item.path.startsWith('/emd/'));
  const cacheViewNames = emdCacheViews.map((item) = > routes.find((route) = > `/emd${route.path}` === item.path)) || [];
  return cacheViewNames.map((name) = >name! .name); });</script>
Copy the code

Consider extracting this as a common component, applying only the prefix.

Ghost rely on

Ghost dependencies are simple to explain when a package is not installed (not in package.json, but the user is able to reference the package). The problem is that after a dependency version is changed, the compiler reports an error and a dependency cannot be found.

This is usually caused by the node_modules structure, such as using YARN to install dependencies on a project that have a dependency called foo, which also depends on bar, Yarn does a flattening of the installed node_modules (as it did after NPM V3), flattening dependencies under node_modules so that foo and bar appear at the same level. So according to nodeJS ‘path-finding principle, the user can require foo as well as require bar.

It is recommended to use PNPM to manage dependencies. It creates a non-flat node_modules directory where code can access only the dependencies set for the current project. It is recommended to restrict installation methods in the project and prohibit non-PNPM installation dependencies:

{
    "scripts": {
        "preinstall": "npx only-allow pnpm"}}Copy the code

conclusion

Above for the front landing of the main problems in the practical project summary, also there are many problems in the current project needs to be improved, such as the redundant state management, build dilemma (Bundle or Bundleless) coverage, automation, etc., we will try to solve the existing problems and continuous iterative evolution, also welcome more relevant experience exchange.

The attachment

  • Micro front landing those things desensitization version
  • Micro front landing those things desensitization version. PPTX

If you want to download PPT, please go to 👍