乾坤微服务的使用

news2025/1/16 4:49:24

前言:

        在这里整理下用乾坤来开发微服务的一些资料。

使用好处:

        使用乾坤可以实现什么效果呢?众所周知,前端的框架五花八门,react/vue/angular等各领风骚,那么如果我们有需要把不同技术栈的项目整合起来,应该怎么去做呢?如果统一技术栈进行开发,工作量太大,成本也高,那还能怎么做呢?没错,这就是我们乾坤技术出来的由来,可以通过他把不同的项目进行一个融会贯通,让他们可以实现亲密的联系,又能各自发展。

乾坤的官网地址:点我

乾坤的逻辑流程:

如何去使用:

1、安装
yarn add qiankun
npm i qiankun -S 
2、主应用的main.js
// (1)导入乾坤框架
import { registerMicroApps, start } from "qiankun";
// (2)注册乾坤子应用
registerMicroApps([
  {
    name:"sub-application01", //子应用名称
    entry:"//localhost:8001", //子应用入库地址
    container:"#container", //主应用容器
    activeRule:"/sub-application01", //主应用路由匹配规则
    props:{
      token:"sub-application-001"
    } //主应用传递参数
  },
  // {
  //   name:"sub-application02",
  //   entry:"//localhost:8002",
  //   container:"#container",
  //   activeRule:"/sub-application02",
  //   props:{
  //     token:"sub-application-002"
  //   }
  // }
]);

 
//(3)开启乾坤
start();
3、主应用的配置  initGlobalState(state)
  • 参数
  • state - Record<string, any> - 必选
  • 用法定义全局状态,并返回通信方法,建议在主应用使用,微应用通过 props 获取通信方法
import { initGlobalState } from 'qiankun';

// 跨应用共享状态
const initialState = {
  hasCallPhone: false, // 是否拨打电话
  outsidePhone: '', // 外部电话号码
  isLocal: true, // 是否是本地号码
  channelId: '', // 渠道
  leadsId: '',
  hasSendMsg: false, // 是否发送短信
  maSend: {}, // MA的leadsId,channelId
  hasSendEmail: false, // 是否发送邮件
  contactHistory: false, // 是否展示联系历史
  customerId: '', // 联系历史需要的id,
  newDict: false, // 是否新增字典
  addDictId: '', // 传入字典id
  callDetails: false, // 是否展示通话详情
  channelSessionId: '', // 通话详情需要的id
  urgentObj: null, // 获取紧急程度
  socketCallback: null,
  taskList: [],
  isCustomerEdit: false, // 是否可以编辑客户
  trendsLayout: [], // 客户表单
  dynamicFields: [], // 动态字段
  fixedFieldsComponent: [], // 固定字段
  operateType: '', // 操作方式,是新增还是编辑
  callerName: '', // 主叫号人员名称
  calledName: '', // 被叫号人员名称
  roomNumber: '', // csp呼叫房间
  softPhone: {
    curOperate: '', // 呼叫状态
    hasSipConnected: false, // 电话连接状态
    mediaAvailabled: false, // 音频媒体
    webrtcConfig: {}, // 初始化连接webrtc配置
  },
  imPageNoticeInfo: {}, // 内部聊天页面通知相关数据
  iqcPageNoticeInfo: {}, // 内部支持页面通知相关数据
  reconnectCallback: null, // 内部支持断网重连回调
  reconnectImCallback: null, // IM
  callVoiceCallback: null,
  callVoiceInfo: {},
  goConversation: false, // 通讯录跳转
};
const actions = initGlobalState(initialState);

export default actions;
4、主应用中手动加载微应用的方式:
import { loadMicroApp } from 'qiankun';
let leftMicroApp = null;

方法内部:
leftMicroApp = loadMicroApp({
  name: 'crm_core',
  entry: '//localhost:9011',
  container: '#wrapper__right',
  props: {
    path: 'customerTabs',
  },
});

//组件销毁,调用子应用的 unmount方法
destroyed() {
  leftMicroApp.unmount()
},
5、子应用中
1、新建文件:public-path.ts/ public-path. js
/* eslint-disable camelcase */
if (window.__POWERED_BY_QIANKUN__) {
  __webpack_public_path__ = window.__INJECTED_PUBLIC_PATH_BY_QIANKUN__
}
2、main.ts/main.js
import './core/public-path'

// vue3中写法
const __qiankun__ = window.__POWERED_BY_QIANKUN__
__qiankun__ || render()



// vue2中写法
//创建子应用渲染函数
function render(props = {}) {
  const { container } = props;
  router = new VueRouter({
    base: window.__POWERED_BY_QIANKUN__ ? '/app-vue/' : '/',
    mode: 'history',
    routes,
  });
 
  instance = new Vue({
    router,
    render: (h) => h(App),
  }).$mount(container ? container.querySelector('#app') : '#app');
};
 
// 独立运行时
if (!window.__POWERED_BY_QIANKUN__) {
  render();
};
3、打包配置,vue.config.js
configureWebpack: {
    output: {
      library: `${name}-[name]`,
      libraryTarget: 'umd', // 把微应用打包成 umd 库格式
      jsonpFunction: `webpackJsonp_${name}`, 
      filename: 'static/js/[hash:8].bundle.js'
    },
  },
6、微应用不需要额外安装任何其他依赖即可接入 qiankun 主应用。

微应用需要在自己的入口 js (通常就是你配置的 webpack 的 entry js) 导出 bootstrap、mount、unmount 三个生命周期钩子,以供主应用在适当的时机调用。

生命周期钩子封装

/**
 * bootstrap 只会在微应用初始化的时候调用一次,下次微应用重新进入时会直接调用 mount 钩子,不会再重复触发 bootstrap。
 * 通常我们可以在这里做一些全局变量的初始化,比如不会在 unmount 阶段被销毁的应用级别的缓存等。
 */
export async function bootstrap() {
  console.log('react app bootstraped');
}
/**
 * 应用每次进入都会调用 mount 方法,通常我们在这里触发应用的渲染方法
 */
export async function mount(props) {
 
}

/**
 * 应用每次 切出/卸载 会调用的方法,通常在这里我们会卸载微应用的应用实例
 */
export async function unmount(props) {
  
}

/**
 * 可选生命周期钩子,仅使用 loadMicroApp 方式加载微应用时生效
 */
export async function update(props) {
  console.log('update props', props);
}

个人项目中用法:

main.ts

import './core/public-path'
import { lifeCycle, render } from './core/life-cycle'

const { bootstrap, mount, unmount } = lifeCycle()
export { bootstrap, mount, unmount }

const __qiankun__ = window.__POWERED_BY_QIANKUN__
__qiankun__ || render()

life-cycle.ts

...
/**
 * 微应用生命周期
 */
const lifeCycle = (): { [key: string]: (props?: qkProps) => Promise<void> } => {
  return {
    async bootstrap(props) {
      console.log(`bootstrap props: ${props}`);
    },
    async mount(props) {
      console.log(`mount props: ${props}`);
      if (props) {
        // 生成动态路由
        const availRoutes = assyAvailRoutes(props.menuLists, 1, "", APP_NAME);
        // 扁平化菜单树
        const flatMenus = flatMenuTree(props.menuLists);
        // 将菜单树、动态路由、扁平菜单树存入全局状态中
        store.commit("user/SET_MENUS", { menuLists: props.menuLists, availRoutes, flatMenus });
        // 将角色列表存入全局状态中
        store.commit("user/SET_ROLES", props.roles);
        store.commit("user/SET_USERINFO", props.userInfo);
        const routes = selfRoutes.concat(availRoutes);
        props.routes = routes;
        store.commit("chat/SET_SINGLE_CONFIG_EVO", []);
        // 如果开启内部聊天语音通话时获取有没有语音聊天权限
        if (matchFuncConfig("INTERNALCHAT_SOFTPHONE_ACTIVE") && store.state.chat.singleConfigEvo.length === 0) {
          getSingleMyConfigs();
        }
        // props.functions.sendOrder({
        //   message: {
        //     type: 'typing',
        //     sendUserId: '',
        //     groupType: ''
        //   }
        // });
        actions.setActions(props);
        actions.setGlobalState({
          socketCallback: (data: any, extraParams: any) => {
            store.commit("chat/SET_SOCKET_MAINAPP_PARAMS", extraParams);
            const { namespace } = extraParams;
            // 接收到父应用分发的消息,进行处理
            if (namespace === "im") {
              if (data.type === spm.ON_PING) {
                imDispatchMessage({ messageType: cmd.SOCKET_PING });
              } else {
                imDispatchMessage({
                  messageType: enumMsg[data.messageType],
                  message: data.message,
                });
              }
            }
            if (namespace === "iqc") {
              if (data.type === spm.ON_PING) {
                iqcDispatchMessage({ messageType: cmd.SOCKET_PING });
              } else {
                iqcDispatchMessage({
                  messageType: enumMsg[data.messageType],
                  message: data.message,
                });
              }
            }
          },
          // 断网重连回调
          reconnectCallback: () => {
            store.commit("internal/SET_RECONNECTED_COUNT");
          },
          // 断网重连回调
          reconnectImCallback: (networkStatus:string) => {
            utilHelper.handleDisconnectOrOnreconnected(networkStatus)
            console.log('##################执行reconnectImCallback',networkStatus);
          },

        });
      }
      await render(props);
    },
    async unmount() {
      // 关闭所有的页面通知实例
      const { pageNoticeInstances = {} } = store.state.chat;
      const instanceKeys = Object.keys(pageNoticeInstances);
      forEach(instanceKeys, (key) => {
        const notifyInstance = pageNoticeInstances[key];
        notifyInstance.close();
      });
      console.log("unmount props");
      instance.unmount();
      instance = null;
      router = null;
    },
    async update(props) {
      console.log(`update props: ${props}`);
    },
  };
};

async function render(props?: qkProps): Promise<void> {
  let basePath = "";
  // 如果是生产环境
  if (process.env.NODE_DEV === "production") {
    // 如果是子应用,使用二级域名前缀,反之使用带internalPortal三级域名
    basePath = __qiankun__ ? `/${APP_NAME}` : `/internalPortal/${APP_KEY}/`;
  } else {
    // 如果非生产环境,并且不是子应用,
    basePath = __qiankun__ ? `/${APP_NAME}` : "/";
  }

  // 初始化固定路由
  let routes = selfRoutes;
  if (__qiankun__) {
    // 如果是微应用,则使用主应用传递的路由集合
    if (props?.routes) routes = props?.routes;
  } else if (store.state.user.accessToken) {
    // 如果没有授权令牌
    // 请求菜单树,非子应用时不控制权限
    const response: AxiosResponse = await axiosSingle(getCompleteTree(), false);
    if (response.data.length > 0) {
      // 获取当前子应用相关的菜单
      let menuLists = response.data[0].children.filter((item: IMenu) =>
        includes(["conversation", "organization"], item.i18n)
      );
      // 递归生成菜单
      menuLists = recurseTree(menuLists, "");
      if (menuLists.length) {
        // 生成动态路由
        const availRoutes = assyAvailRoutes(menuLists, 1, "", APP_NAME);
        // 扁平化菜单树
        const flatMenus = flatMenuTree(menuLists);
        // 将菜单树、动态路由、扁平菜单树存入全局状态中
        store.commit("user/SET_MENUS", { menuLists, availRoutes, flatMenus });
        // 叠加固定路由和动态路由
        // routes = selfRoutes.concat(availRoutes)
        selfRoutes[0].children = availRoutes;
        routes = selfRoutes;
      }
    }
  }
  router = createRouter({
    history: createMemoryHistory(basePath),
    routes,
  });
  instance = createApp(App).use(router).use(store).use(i18n).use(plugin, { imports: [] });

  // 全局注册El组件
  components.forEach((item) => {
    if (item) instance.use(item);
  });
  // 全量导入El图标
  for (const key in Icons) {
    if (Reflect.has(Icons, key)) {
      instance.component(key, Icons[key]);
    }
  }
  // 注册按钮授权指令
  instance.use(authDirective);
  // 注册按钮授权全局方法
  instance.config.globalProperties.$vAuth = function (key: any) {
    return directiveAuth(this, key);
  };
  instance.use(draggable);

  instance.mount(props?.container ? props.container.querySelector("#appInternalChat") : "#appInternalChat");
  // instance.use(VueVirtualScroller);
  // instance.component('DynamicScroller', VueVirtualScroller.DynamicScroller)

  // 前置路由守卫
  router.beforeEach(async (to: any, from: any) => {
    if (!__qiankun__) {
      // 1 如果不是子应用
      if (store.state.user.accessToken) {
        if (!store.state.user.userInfo) {
          const infoConfig = configRequest(`${GlobalConfig.API_HRMS_URL}/employee/current`, httpMethod.GET);
          const response1 = await axiosSingle(infoConfig);
          const userInfo = response1.data;
          store.commit("user/SET_USERINFO", userInfo);
          // 1.1 如果有授权令牌
          if (to.path === "/login") {
            // 1.1.1 如果访问页面为登录页,则跳转到首页
            return "/";
          } else if (to.matched.length) {
            // 1.1.2 如果有匹配的路由,则进行跳转
            return true;
          } else {
            // 1.1.3 如果找不到匹配路由,则跳转到未授权报错页面
            // next({ path: '/403', replace: true })
            return false;
          }
        }
      } else if (to.path === "/login" && to.query.code) {
        // 1.2 如果没有令牌并跳转到登录页,并有授权码
        return true;
      } else {
        // 如果没有令牌并且没有授权码,则跳转到sso进行登录
        signIn();
      }
    } else if (to.matched.length) {
      // 2 如果是子应用,并且有匹配的路由,则进行跳转
      return true;
    } else {
      // 3 如果没有匹配路由,则跳转到未授权报错页面
      // next({ path: '/403', replace: true })
      return false;
    }
  });
}

export { lifeCycle, render };

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1838903.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

为何Proteus用户争相拥抱SmartEDA?揭秘背后的强大吸引力!

在电路设计与仿真领域&#xff0c;Proteus一度以其稳定性能和丰富功能赢得了众多用户的青睐。然而&#xff0c;近年来&#xff0c;越来越多的Proteus用户开始转向SmartEDA&#xff0c;这一新兴电路仿真软件正迅速崭露头角&#xff0c;成为行业内的翘楚。那么&#xff0c;究竟是…

MySQL数据库的列类型

数值 tinyint 十分小的数据 1个字节 smallint 较小的数据 2个字节 mediumint 中等大小的数据 3个字节 int 标准的整数 4个字节&#xff08;常用&#xff09; bigint …

ARM架构简明教程

目录 一、ARM架构 1、RISC指令集 2、ARM架构数据类型的约定 2.1 ARM-v7架构数据类型的约定 2.2 ARM-v8架构数据类型的约定 3、CPU内部寄存器 4、特殊寄存器 4.1 SP寄存器 4.2 LR寄存器 4.3 PC寄存器 二、汇编 1、汇编指令&#xff08;常用&#xff09; 2、C函数的…

屏蔽房是做什么用的?为什么需要定期检测?

屏蔽房对于不了解的人来说&#xff0c;可能光看名字不知道是做什么的&#xff0c;但是对于一些企业或者机构&#xff0c;却是再熟悉不过的了。和名字一样&#xff0c;屏蔽房是对空间内的信号以及一些外界环境条件进行隔绝&#xff0c;在一些有特殊要求的企业机构中&#xff0c;…

刚刚发布!这4本期刊已剔除SCI收录,附完整目录下载

科睿唯安于6月19日更新了SCIE、SSCI、AHCI、ESCI四大数据库最新收录期刊目录。 2024年第一版——2024年1月24日更新 2024年第二版——2024年2月19日更新 2024年第三版——2024年3月18日更新 2024年第四版——2024年4月15日更新 2024年第五版——2024年5月20日更新 2024年…

疯狂买买买!你的支付环境真的安全吗?

在日常生活中&#xff0c;移动电话为我们带来了更多的方便。然而&#xff0c;我们在享受手机支付的便捷之余&#xff0c;也应充分认识到风险&#xff0c;增强防范意识&#xff0c;慧眼识诈。 小亿提醒&#xff1a;大家在购物之余&#xff0c;务必要注意手机支付的安全性&#…

二维码分班查询系统你还不会用?

分班查询系统&#xff0c;已经成为许多学校管理分班流程的得力助手。当新学期伊始&#xff0c;学校需要进行分班&#xff0c;而传统的手工分班方式不仅耗时&#xff0c;还容易出错。这时&#xff0c;一个智能的分班查询系统就显得尤为重要。 作为老师&#xff0c;您可能已经意识…

电脑开机黑屏怎么办?教你3招轻松解决

电脑开机黑屏是一种常见但令人沮丧的问题。无论是台式机还是笔记本电脑&#xff0c;用户都可能遇到这种情况&#xff0c;导致无法正常使用电脑。黑屏问题可能由多种原因引起&#xff0c;包括硬件故障、软件冲突、驱动问题等。本文将介绍电脑开机黑屏怎么办的三种方法&#xff0…

Geoserver源码解读三 GeoServerBasePage

一、概述 org.geoserver.web.GeoServerBasePage 类&#xff0c;在Geoserver中是所有页面类的基类&#xff0c;也是单独存在的一个主UI界面入口文件。拿到源码后可以在里面进行肆意的魔改&#xff0c;也可以单独创建一个工程写根据它扩展。下面以登录的代码作为切入点&#xff0…

CarService的构成和初始化分析

以下分析&#xff0c;基于安卓13的AAOS。 代码构成 packages/services/Car CarService相关代码&#xff0c;主要是在这个目录下 frameworks/opt/car/services 主要是carservice启动相关。 其它目录&#xff1a;audio_policy_configuration.xml和car_audio_configuration.xm…

<Rust><iced>基于rust使用iced构建GUI实例:如何将svg格式转为ico格式图片?

前言 本专栏是Rust实例应用。 环境配置 平台:windows 软件:vscode 语言:rust 库:iced、iced_aw 概述 本文是专栏第4篇实例,依旧是一个图像格式转换程序,基于rust的svg库resvg、图像处理库image以及文件处理库rfd。 流程是先用resvg获取svg图片的数据并将其转为png数据…

llama-factory微调chatglm3

一、定义 案例/多卡 二、实现 案例 1. 下载chatglm3-6b-32k模型 2. 配置数据集微调指令 CUDA_VISIBLE_DEVICES0,1 llamafactory-cli train \--stage sft \--do_train True \--model_name_or_path /home/chatglm3-6b-32k \--finetuning_type lora \--template chatglm3 \--d…

零基础入门学用Arduino 第四部分(三)

重要的内容写在前面&#xff1a; 该系列是以up主太极创客的零基础入门学用Arduino教程为基础制作的学习笔记。个人把这个教程学完之后&#xff0c;整体感觉是很好的&#xff0c;如果有条件的可以先学习一些相关课程&#xff0c;学起来会更加轻松&#xff0c;相关课程有数字电路…

nodejs爬取小红书图片

昨天的文章已经描述了可以抓取评论区内容&#xff0c; 抓取图片内容和抓取评论区的内容基本一致 我们可以看到接口信息中含有图片链接&#xff0c;我们要做的就是爬取图片链接然后下载 这边要用到的模块为const downloadrequire(download) 将爬到的图片链接存放到images数组…

STM8单片机的GPIO口介绍

本篇文章依托于stm8单片机和lora模块 目录 一、GPIO口测试相关硬件电路图 &#xff08;a&#xff09;USB转串口底板PA3控制LED6 (b)Lora模块底板PA3接口 &#xff08;c&#xff09;LED灯电路 二、参考官方例程实现GPIO口的输出 三、GPIO相关函数的了解 &#xff08;1&a…

docker安装使用

文章目录 docker产生的原因传统虚拟机容器化技术 docker组成安装docker镜像加速docker安装过程中遇到的问题以及解决办法Errors during downloading metadata for repository root_:Failed to set locale, defaulting to C.UTF-8 docker产生的原因 传统虚拟机 在不使用docker…

从“野人饭”走红,探索品牌户外化营销趋势丨小红书内容分析

wildeat&#xff0c;户外是人的天性的回归 近来&#xff0c;“wildeat&#xff08;户外野吃&#xff09;”的风潮在小红书逐渐兴起。越来越多的人选择到户外吃一顿&#xff0c;做一次“野人”&#xff0c;主打一个只要氛围到了&#xff0c;就地开饭&#xff0c;不愁吃什么&…

AI智能写作工具, 免费在线智能创作内容网站

对于需要创作内容的同学&#xff0c;选择一款适合自己的AI写作工具可以极大的提高创作效率。下面小编就来和大家分享几款可以生成高质量原创内容的AI写作工具。 1. Kimi智能助手 Kimi智能助手是一款集成了先进算法的AI工具&#xff0c;它能够理解复杂的语言模式&#xff0c;生…

如何用Vue3和ApexCharts打造引人注目的3D径向条形图

本文由ScriptEcho平台提供技术支持 项目地址&#xff1a;传送门 使用 ApexCharts 构建美观的 Vue.js 径向条形图 应用场景 径向条形图是一种用于可视化单一数据点及其与目标或理想值的关系的图表类型。它在显示进度、完成率或其他类似度量时非常有用。 基本功能 这段代码…

划分子网和构造超网的学习

子网掩码长度&#xff1d;32位 某位&#xff1d;1&#xff1a;IP地址中的对应位为网络号和子网号 某位&#xff1d;0&#xff1a;IP地址中的对应位为主机号 从一个 IP 数据报的首部并无法判断源主机或目的主机所连接的网络是否进行了子网划分。 使用子网掩码(subnet mask)可…