大型 Web 应用插件化架构探索

简介: 随着 Web 技术的逐渐成熟,越来越多的应用架构趋向于复杂,例如阿里云等巨型控制台项目,每个产品下都有各自的团队来负责维护和迭代。不论是维护还是发布以及管控成本都随着业务体量的增长而逐渐不可控。在这个背景下微前端应用而阿里云和双十一有什么关系生,微前端在阿里内部已经有许多成熟的实践,这里不再赘述。本文以微前端为引子(蹭热度),探讨一些另类的 Web 应用所面临的类似问题。

前言

随着 Web 技术的逐阿里云双十一渐成阿里巴巴双十一市场活动熟,越来越多的应用架构趋向于复杂,例如阿里云等巨型控制台项目,每个产品下都有各自的团队来负责维护和迭代。不论是维护还是发布以及管控成本都随着业务体阿里云代金券可以赠送吗量的增长而逐渐不可控。阿里云在这个背景双11阿里云阿里云优惠微前端应用而生,微前端阿里云续费优惠券在阿里内部已经有许多成熟的实阿里云优惠券领取践,这里不再赘述。本文以微前端为引子,阿里会员日和双十一探讨一些另类的 Web 应用所面临阿里巴巴双十一市场活动的类似问题。

现代文本编辑器沉浮

2018年微软 GitHub 后,Atom便经常被拿来调侃,所谓一山不容二虎。在 VS Code 已经成为一众前端工程师编辑器首选的当下,Atom 的地位显得很尴尬,论性能被同为 Electron 的 VS Code 秒杀,论插件,VS Code 去年插件总数就已经突破 1w 大关,而早发布一年多的 Atom 至今还停留在 8k +。再加上微软官方主导的 LSP/DAP 等重量级协议的普及,时至今日 Atom 作为曾经 Web/Electron 技术标杆应用的地位早已被 VS Code 斩落马下。

大型 Web 应用插件化架构探索

网上关于 Ato阿里云双十一活动拼团m 的日渐衰落的讨论,始终离不开性能。Atom 的确太慢了,究其原因很大程度上是被其插件架构所拖累的。尤其是 Atom 在 UI 层面开放过多的权限给插件开发者定制,插件质量良萎不齐以及 UI 完全开放给插件后带来的安全隐患都成为 Atom 的阿喀琉斯之踵。甚至其主界面的阿里云服务器双11活动 Fil阿里云双十一eTree、Tab 栏、Setting Vi阿里云双11优惠ews 等重要组件都是通阿里云拼团活动过插件实现的。相比之下 VS Code 则封闭很多,VS Code 插件完全运行在 Node阿里云双十一活动老用户.j阿里云双十一服务器s 端,对于 UI 的定制性只有极个别被封装为纯方法调用的 A阿里云双11活动PI。

大型 Web 应用插件化架构探索

但另一方面,VS Code 这种相对封闭的插件 UI 方案,一些需要更强定制性的功能便无法满足,更多插件开阿里云双十一红包发者开始魔改 VS Code 底层甚至源码来实现定制。例如社区很火的 VS Code Background,这款插件通过强行修改 VS Code 安装文件中的 CSS 来实现阿里巴巴双十一活动目的编辑器区域的阿里云双十一活动地址背景图。而另一款 VSC Netease Music 则更激进,因为 VS Code 捆绑包中的 Electron 剔除了 FFmpeg 导致在 Webview 视图下无法播放音视频,使用此插件需要自行替换 FFmpeg 的动态链接库。而这些插件不免会对 VS Code 安装包造成一定程度的破坏,导致用户需要卸载重装。

不止编辑器 - 飞个马

Fi阿里云几折优惠gma 是一个在线协作式 UI 设计工具, 相比 Sketch 它具有跨平台、实时协作等优点,近年来逐渐受到 UI 设计师们的青睐。而近期 Figma 也正式上线了其插件系统。

大型 Web 应用插件化架构探索

作为一个 Web 应用,Figma 的插件系统自然也是基于 JavaScript 构建的,这一定程度上降低了阿里云过了优惠期多少钱开发门槛。自去年6月份 Figma 官方宣布开放插件系统阿里云双十一红包测试以来,已经有越来越多的 Designner/De阿里巴巴双十一市场活动velope阿里云和你拼了r 开发阿里云双十一了300+ 插件,其中包括阿里云双十一图形资源、文件归档、甚至是导入 3D 模型等。

大型 Web 应用插件化架构探索

Figma 的插件系统是如何工作的?

这是一个基于 TypeScript + React 技术栈,使用 Webpack 构建的 Figma 插件目录结构

.阿里云优惠购买
├── README.md
├── figma.d.ts
├── manifest.js阿里云双十一2019on
├阿里云服务器双十一特惠── package-lock.json
├── package.json
├── src
│   ├── code.ts
│   ├── logo.svg
│   ├── ui.css
│   ├── ui.html
│   └── u阿里云优惠券怎么使用i.tsx
├── tsconfig.json
└── webpack.c阿里云服务器双11活动onfig.js

在其 manifest.json 文件中包含了一些简单的信息。

{
"name": "React Sample",
"id": "738168449509241862",
"api": "1.2018双11活动0.0",云服务器双11活动
"main": "阿里云优惠购买dist/code.js",
"ui": "di阿里双11现在交易额st/ui.html"
}

可以看出 Figma 将插件入口分为了 main 与 ui 两部分, main 中包含了插件实际运行阿里云服务器活动时的逻辑,而 ui 则是一个插件的 HTML 片段。即 UI 与逻辑分离。安装一个Color Sear阿里双十一活动ch 插件后观察页面结构可以发现 main 中的 js 文件被包裹在一个 iframe 里加载到页面上,阿里云新人优惠券关于 main 入口的沙箱机阿里云活动制后文中阿里云服务器双11活动有详细的阐述。而 ui 中的 HTML 最终也被包裹在一个 if阿里云双11活动rame 里渲染出来,这将有效的避免插件 UI 层 CSS 代码导致全局样式污染

[图片上传失败...(image-82d0bb-1615950533867)]

Figma Developers 文档中 有一章节 How Plugins Run 对其插件系统运行机制进行了简单的介绍,简单来说 Figma 为插件中逻辑层的 main 入口创建了一个最小的 JavaS双11活动阿里云的作用cript 执行环境,它运行在浏览器主线程上,在这个执行环境中插件代码无法访问到一些浏览器全局的双十一阿里云有优惠吗 API,从而也就无法在代码层阿里云双十一活动老用户面对 Figma 本身运行造成影响。而 UI 层有且仅有一份 HTML 代码片段,在插件阿里云双十一2020被激活后被渲染到一个弹窗中。

[图片上传失败...(image-c17936-161595053阿里云代金券领取3867)]

Figma 官方博客中对其插件的沙箱机制做了详细的阐述。起初他们尝试的方案阿里双11数据是 iframe,一个浏览器自带的沙箱环境。阿里云活动将插件代码由 iframe 包裹起来,由于 iframe 天然的限制,这将确保插阿里云双十一红包件代码无法操作 Figma 主界面上下文阿里云优惠,同时也可以只开放一份白阿里云双十一名单 API 供插件调用。乍一看似乎解决了问阿里云优惠券领取题,但由于 iframe 中的插件脚本只能通过 postMessage 与主线程通信,这导致插件中的任何 API 调用都必须被包装为一个异步 async/await 的方法,阿里云双十一2019这无疑对 Figma 的目标用户非专业前端开发者的设计师不够友好。阿里云优惠卷其次对于较大的文档,postMessage 通信序列化的性能成本过高,甚至会导致内存泄漏。

Figma 团队选择回到浏览器主线程,但直接将第三方代码运行在主线程,由此引发的安全问题是不可避免的。最终阿里云双11优惠他们发阿里云服务器拼团活动现了一个阿里云优惠尚在 s阿里云代金券可以赠送吗tage2 阶段的草案 Realm API。Re阿里云双11优惠alm 旨在创建一个领域对象,用于隔离第三方 JavaScript 作用域的 API阿里云双11优惠

let g = window; // outer global
let r = new Realm推广阿里云双十一赚10万(); // root realm
let f = r.evaluate("(function() { return 17 })");
f() === 17 /阿里云双11续费/ true
Reflect.getPrototypeOf(f) === g.Function阿里云代金券阿里云优惠券.prototype // false
Reflect.getPrototypeOf(f) =阿里云双11续费== r.globalThis.Function.prototype // true

值得注意的是,Realm 同样可以使阿里云拼团用 JavaScript 目前已有的特性来实现,即 with 与 Proxy。这也是目前社区比较流行的沙箱方案。

const whitelist = {
wind阿里双11现在交易额iw: undefined,
document: undef阿里巴巴双十一营销活动ined,
console: window.console,
};
const sc阿里云代金券领完不见了opeProxy = new Proxy(whitelist, {
get(target, prop) {
if (prop in target) {
retu阿里云双11rn target[prop]
}
return undefined
}
});
with (scopeProxy) {
eval("console阿里云优惠.log阿里云双十一活动地址(document.write)") // Cannot read property 'write' of undefined!
ev双11阿里云al("console.log('hello')")        // hello
}

前文中 Figma 插件被 iframe 所包裹的插件 main 入口即包含了一个被 Realm 接管的作用域,你可以认为是类似这段示例代码中的一阿里云双十一优惠份 白名单 API,毕竟维护一份白名单比屏蔽黑名单实现起来更简洁。但事实上由于 JavaScript 的阿里云双十一原型式继承,插件仍然可以通过 console.log 方法的原型链访问到外部对象,理想的解决方案是将这些白名单 API 在 Realm 上下文中包装一次,从而彻底隔离原型链。

const safeLogFactory = realm.evaluate(`
(func阿里云双十一卡住了吗tion safeLogFact阿里云双十一ory(unsafe阿里云续费优惠券Log) {
return function safeLog(...args) {
unsafeLog(...args);
}
})
`);
const safeLog = safeLogFactory(console.log);
const outerIntrinsics = safeLog i阿里云领代金券nstanceOf Function;
const innerIntrinsics = realm.evaluate(`log instanceOf Function`, { log: safeLog });
if (outerIntrinsics || !阿里云服务器双11活动innerIntrinsics) throw new TypeError();
realm.evaluate(`log("Hello outside world!")`, { log: safeLog });
大型 Web 应用插件化架构探索

显然为每一个白名单中的 API 做这样操作的阿里云双11续费工作是非常繁杂且容易出错的。那么如何构建一个安全且易于添加 API 的沙阿里云拼团活动箱环境呢?

Duktape 是一个由 C++ 实现的用阿里云双十一活动2018于嵌入式设备的 JavaScript 解释器,它不支持任何浏览器 API,自然地它可以被编译到阿里云服务器拼团活动 WebAssembly双11阿里云,Figma 团队将 Duktape 嵌入到 Realm 上下阿里云双11活动文中,插件最终通过 D云服务器优惠活动uktape 解释执行。这样可以安全的实现插件所需 API,且不用担心插件会通过原型链访问到沙箱外部。

[图片上传失败...阿里双11数据(image-630b63-161595053386阿里云和双十一有什么关系7)]

这是一种被称为 Membrane Pattern 的防御性的编程模式,用于在程序中与子组件(广义上)实现一层中介。阿里云双十一活动拼团简单来说阿里云优惠券就是代理(Proxy),为一个对象创建一个可控的访问边界,使得它阿里云服务器双11活动可以保留一部分特阿里双11现在交易额性给第三方嵌入脚本,而屏蔽一部分阿里活动推荐不希望被访问到的特性。关于 Membrane阿里云双十一活动 的详细论述可以查看 Isolating application sub-components with membranes 与 Membranes in JavaScript 这两篇文章。

大型 Web 应用插件化架构探索

这是最终 Figma 的插件方案,它运行在主线程,不需要担心 postMessage 通信带来的传输损耗。多了一次 Duktape 解释执行的消耗,但得益于 WebA阿里云双十一活动sse阿里云代金券如何使用mbly 出色的性能,这部分消耗并不是很大。

另外 F阿里云双十一活动拼团igma 还保留了最初的 iframe ,允许插件可以自行创建 iframe ,并在其中插入任意 JavaScript ,同时它可以与沙箱中的 JavaScript 脚本通过 postMessage 相互通信。

鱼和熊阿里云优惠活动掌如何兼得?

我们把这类插件的需求总结为在 Web 应用中运行第三方代码及其自定义控件,它有与开头提到的微前端架构非常相似的一些问题。

  1. 一定程度上的 JavaScri阿里云服务器拼团活动pt 代码沙箱隔离机制,应用主阿里云服务器双11活动体对第三方代码(或子应用)有一定的管控能力
  2. 样式强隔离,第三方代码样式不双11活动阿里云的作用对应用主体产生 CSS 污染

JavaScript 沙阿里云续费优惠券

JavaScript 沙箱隔离在社区是个经久不衰的话题,最简单的 iframe 标签 Sandbox 属性就已经能做到 JavaScript阿里云代金券怎么用 运行时的隔离,社区较为流行的是利用一些语言特性(with、realm、Proxy 等 API )阿里云和双十一有什么关系屏蔽(或代理) Window、Document 等全局对象,建立白名单机制,对可能潜在危险操作的 API 重写(如阿里云 Console OS - Browser VM)。另外还有 Figma 这种尝试嵌入平台无关的 JavaScript 解释器,所有第三方代码都通过嵌入的解释器来执行。以及阿里云双11活动利用 Web Worker 做 DOM Diff 计算,并将计算结果发送回 UI 线程来进行渲染,这个方案早在 2013 年就已经有人进行了实践,这篇论文中作者将 JSDOM 这一 Node.js 平台广泛流行的测试库运行在 Web Worker。而近些年来也有 preact-wor双11阿里云ker-demo 、react-worker-dom 等项目基于 Web Wo阿里云服务器双11活动rker 的 DOM Renderer 尝试将 DOM API 代理到 Worker 线程。而 Google AM阿里云服务器双十一特惠P Project阿里云 在JSCONF 2018 US 对外公布的 worker-dom 则将 DOM API 在 Web Worker阿里云代金券领完不见了 端实现了 DOM API,虽然实践下来还存在一些问题(例如同步方法无法模拟),但 Wor阿里云优惠券领取kerDOM 在性能和隔离性上都取得阿里云服务器双11活动了一定成果。

大型 Web 应用插件化架构探索

以上这些解决方案被广泛的应用在各种插件化架构的 Web 应用中,但大多都是 Case By Cas阿里双11现在交易额e,每种解决方案都有各自的成本与取舍。

CSS 作用域

CSS 样式隔离方案中,如上文中 Figma阿里双十一活动 使用 iframe 渲染插件界面,牺牲一部分性能换来了相对阿里巴巴双十一市场活动完美的样式隔离。而在现代前端工程化体系下,可以通过 CSS Module 在转译时对 class 添加 hash 或 namespace 等方式实现,这类方案较为依赖插件代码编译过程。而更新潮的是利用 Web Component阿里云双十一优惠 的 Shadow DO阿里云双十一活动时间M,将插件元素用 Web Component 包裹起来,Sh阿里云双十一活动拼团adow Root 外部样式无法作用于内部,同样 Shadow R阿里云双十一红包oot 内部的样式也无法影响到外部。

[图片上传失败...(image-28b3bc-1615950533867)]

最后

本文列举了目前编辑器、设计工具这阿里云双11优惠类大型 Web 应用插件化架构下所面临的的一些问题,以及社区实践的解决方案。不论是让人又爱又恨的 iframe ,还是 Realm、Web Worker双11阿里云 、 Shadow DOM 等,目前来说每种方案都有各自的阿里双十一活动优势与不足。但随着 Web阿里云服务器双十一特惠 应用的复杂度增长阿里云优惠码,插件化这阿里双11现在交易额一需求也阿里云200元代金券逐渐被各大标准化组织所重视起来。下一篇将着重介绍 KAITIAN IDE 中插件架构的探索与实践,包阿里云和你拼了括 JavaScri阿里云双11pt 沙箱、CSS 隔离、Web Worker 等。

作者:开发者小助手_LS