plugin-factory.ts 7.8 KB
Newer Older
1 2 3
import { RuntimeError } from '@ibiz-template/core';
import {
  IPluginFactory,
4
  IPluginItem,
5 6 7 8
  RemotePluginConfig,
  RemotePluginItem,
} from '@ibiz-template/runtime';
import { IPSAppPFPluginRef, IPSSysPFPlugin } from '@ibiz/dynamic-model-api';
9
import path from 'path-browserify';
10
import Vue, { VueConstructor, PluginObject } from 'vue';
11

12 13
const { join } = path;

14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60
/**
 * 插件工具类
 *
 * @author chitanda
 * @date 2022-10-21 16:10:29
 * @export
 * @class PluginFactory
 */
export class PluginFactory implements IPluginFactory {
  /**
   * 是否为 http || https 开头
   *
   * @author chitanda
   * @date 2022-11-07 14:11:28
   * @protected
   */
  protected urlReg = /^http[s]?:\/\/[^\s]*/;

  /**
   * 是否已经加载过文件缓存
   *
   * @author chitanda
   * @date 2022-10-31 14:10:17
   * @protected
   * @type {Map<string, boolean>}
   */
  protected cache: Map<string, boolean> = new Map();

  /**
   * 本地开发测试包,只在本地开发生效
   *
   * @author chitanda
   * @date 2022-11-02 21:11:41
   * @protected
   */
  protected devPackages: Map<string, () => Promise<unknown>> = new Map();

  /**
   * 插件缓存
   *
   * @author chitanda
   * @date 2022-10-31 14:10:28
   * @protected
   * @type {Map<string, RemotePluginItem>}
   */
  protected pluginCache: Map<string, RemotePluginItem> = new Map();

61 62 63 64 65 66 67 68 69 70
  /**
   * 预定义插件集合
   *
   * @author chitanda
   * @date 2023-03-09 17:03:46
   * @protected
   * @type {Map<string, IPluginItem>}
   */
  protected predefinedPlugins: Map<string, IPluginItem> = new Map();

71 72 73 74 75 76 77 78 79
  /**
   * 在预置加载清单的插件
   *
   * @author chitanda
   * @date 2023-04-02 23:04:40
   * @type {string[]}
   */
  readonly appPlugins: string[] = [];

80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97
  /**
   * 注册视图默认插件
   *
   * @author chitanda
   * @date 2023-02-06 21:02:10
   * @param {IPluginItem} plugin
   */
  registerPredefinedPlugin(plugin: IPluginItem): void {
    this.predefinedPlugins.set(plugin.name, plugin);
  }

  /**
   * 给入应用实例,将已经加载的过插件注入。主要用于多实例的情况
   *
   * @author chitanda
   * @date 2023-02-02 16:02:51
   * @param {App} app
   */
98
  register(_app: VueConstructor): void {}
99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125

  /**
   * 加载预置插件
   *
   * @author chitanda
   * @date 2023-03-09 18:03:48
   * @param {string} name
   * @return {*}  {Promise<void>}
   */
  async loadPredefinedPlugin(name: string): Promise<void> {
    if (this.predefinedPlugins.has(name)) {
      const plugin = this.predefinedPlugins.get(name)!;
      if (plugin) {
        await this.loadPluginRef(plugin.name, plugin.path);
      }
    }
  }

  /**
   * 插件刚加载完成回来,设置到目前所有的 vue 实例当中
   *
   * @author chitanda
   * @date 2023-02-02 17:02:38
   * @protected
   * @param {PluginObject<unknown>} code
   */
  protected setPluginCode(code: PluginObject<unknown>): void {
126
    Vue.use(code);
127 128
  }

129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152
  /**
   * 设置开发插件,用于本地调试
   *
   * @author chitanda
   * @date 2022-11-02 21:11:56
   * @param {string} name
   * @param {() => Promise<unknown>} fn
   */
  setDevPlugin(name: string, fn: () => Promise<unknown>): void {
    this.devPackages.set(name, fn);
  }

  /**
   * 加载插件
   *
   * @author chitanda
   * @date 2022-10-31 14:10:13
   * @param {IPSSysPFPlugin} plugin
   * @return {*}  {Promise<boolean>}
   */
  async loadPlugin(plugin: IPSSysPFPlugin): Promise<boolean> {
    if (plugin.runtimeObject === true) {
      const pluginRef = plugin as unknown as IPSAppPFPluginRef;
      if (pluginRef) {
153 154 155 156 157 158
        try {
          await this.loadPluginRef(
            pluginRef.rTObjectName,
            pluginRef.rTObjectRepo,
          );
        } catch (error) {
159
          ibiz.log.error(error);
160
        }
161 162 163 164 165 166
      }
    }
    return false;
  }

  /**
167
   * 加载应用插件
168 169 170 171 172 173
   *
   * @author chitanda
   * @date 2022-10-31 16:10:57
   * @param {IPSAppPFPluginRef} pluginRef
   * @return {*}  {Promise<boolean>}
   */
174 175 176 177 178
  async loadPluginRef(
    rtObjectName: string,
    rtObjectRepo: string,
  ): Promise<boolean> {
    if (this.pluginCache.has(rtObjectName)) {
179 180
      return true;
    }
181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198
    let configData: unknown = null;
    const fn = this.devPackages.get(rtObjectName);
    if (fn) {
      configData = await fn();
    } else {
      const pluginPath: string = rtObjectRepo;
      const configUrl = this.urlReg.test(pluginPath)
        ? `${pluginPath}/package.json`
        : `${ibiz.env.pluginBaseUrl}/${join(pluginPath, 'package.json')}`;
      const res = await ibiz.net.axios({
        method: 'get',
        headers: { 'Access-Control-Allow-Origin': '*' },
        url: configUrl,
      });
      if (res.status !== 200) {
        throw new Error(`配置加载失败`);
      }
      configData = res.data;
199 200
    }
    const remotePlugin = new RemotePluginItem(
201 202 203
      rtObjectName,
      rtObjectRepo,
      configData as RemotePluginConfig,
204 205 206 207
    );
    if (remotePlugin) {
      try {
        await this.loadScript(remotePlugin);
208
        this.pluginCache.set(rtObjectName, remotePlugin);
209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226
        return true;
      } catch (error) {
        ibiz.log.error(error);
      }
    }
    return false;
  }

  /**
   * 加载插件
   *
   * @author chitanda
   * @date 2022-11-02 14:11:31
   * @protected
   * @param {RemotePluginItem} remotePlugin
   * @return {*}  {Promise<void>}
   */
  protected async loadScript(remotePlugin: RemotePluginItem): Promise<void> {
227 228 229 230 231 232 233 234 235 236 237 238 239 240
    const pluginPath: string = remotePlugin.repo;
    const { name, version, system, module } = remotePlugin.config;
    let scriptUrl = '';
    if (this.devPackages.has(name)) {
      scriptUrl = join(`${name}@${version}`, module);
    } else if (ibiz.env.dev) {
      scriptUrl = this.urlReg.test(pluginPath)
        ? `${pluginPath}/${join(module)}`
        : `${ibiz.env.pluginBaseUrl}/${pluginPath}/${join(module)}`;
    } else {
      scriptUrl = `${pluginPath}${pluginPath.endsWith('/') ? '' : '/'}${join(
        system,
      )}`;
    }
241 242 243 244 245 246 247 248 249 250 251
    if (scriptUrl) {
      if (this.cache.has(scriptUrl)) {
        return;
      }
      let data: IParams | null = null;
      const url = this.parseUrl(scriptUrl);
      // 本地开发模式下的包引用加载
      if (this.devPackages.has(name)) {
        const fn = this.devPackages.get(name)!;
        // eslint-disable-next-line @typescript-eslint/no-explicit-any
        const pluginModule = (await fn()) as any;
252
        this.setPluginCode(pluginModule.default);
253
        this.cache.set(scriptUrl, true);
254 255
      } else if (ibiz.env.dev) {
        data = await import(scriptUrl /* @vite-ignore */);
256
      } else {
257 258 259 260 261 262 263 264 265 266
        System.addImportMap({
          imports: {
            [name]: url,
          },
        });
        data = await System.import(name);
      }
      if (data) {
        if (data.default) {
          this.setPluginCode(data.default);
267
        } else {
268 269 270
          throw new RuntimeError(
            `远程插件加载失败, 远程插件未找到[default]默认导出`,
          );
271
        }
272 273 274 275 276
        this.cache.set(scriptUrl, true);
      } else {
        throw new RuntimeError(
          `远程插件加载失败, 未找到文件或文件内容格式不正确`,
        );
277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293
      }
    }
  }

  /**
   * 编译请求文件地址
   *
   * @author chitanda
   * @date 2022-10-31 14:10:19
   * @protected
   * @param {string} script
   * @return {*}  {string}
   */
  protected parseUrl(script: string): string {
    if (this.urlReg.test(script)) {
      return script;
    }
294
    let url: string = '';
295 296
    if (this.urlReg.test(ibiz.env.pluginBaseUrl)) {
      if (script.startsWith('/')) {
297 298 299
        url = ibiz.env.pluginBaseUrl + script;
      } else {
        url = `${ibiz.env.pluginBaseUrl}/${script}`;
300
      }
301 302 303 304 305 306
    } else {
      url = `${join(ibiz.env.pluginBaseUrl, script)}`;
    }
    const { origin, pathname } = window.location;
    if (pathname.endsWith('/') && url.startsWith('/')) {
      url = url.substring(1);
307
    }
308 309
    url = `${origin}${pathname}${url}`;
    return url;
310 311
  }
}