低代码系列——js沙箱设计(一)

简介: 由于自己参与过低代码平台开发,所以希望能把我自己开发低代码中遇到的问题或者一些设计思路进行总结汇总,这是开始写的第一篇,也是比较基础的一篇,关于低代码平台的介绍会放在介绍篇章,这篇就不做过多介绍。

简介


由于自己参与过低代码平台开发,所以希望能把我自己开发低代码中遇到的问题或者一些设计思路进行总结汇总,这是开始写的第一篇,也是比较基础的一篇,关于低代码平台的介绍会放在介绍篇章,这篇就不做过多介绍。


这里为什么会一开始介绍js沙箱设计呢?


因为低代码平台,会运行用户本身自己编写的业务逻辑代码,这里就需要平台去运行用户写的js代码,但是js代码保存到数据库是一个字符串,那么平台应该怎么运行呢?


答案是js沙箱,那么如何设计一个沙箱呢?按照低代码平台的需要特性,主要以下几方面:

  • 隔离,隔离是为了保证当前执行代码不影响整个平台的代码
  • 插入,沙箱允许插入平台的内置对象
  • 容错,沙箱内代码即使有错误,也不影响整个平台执行

沙箱


在设计沙箱之前,我们先对沙箱有个了解:

在计算机安全中,沙箱(Sandbox)是一种用于隔离正在运行程序的安全机制,通常用于执行未经测试或不受信任的程序或代码,它会为待执行的程序创建一个独立的执行环境,内部程序的执行不会影响到外部程序的运行。

通俗的讲,就是由我们主程序自己设定一个区域,用来执行代码,且这段代码如何执行都不会影响到外部的主程序。


举几个我们开发中经常会用的沙箱:

  • Vue template里的表达式,如:

    {{ 1+1 }}

    ,执行1+1就是Vue设计的一个沙箱机制
  • 开发Chrome插件,插件里的代码有很多限制条件,循序Chrome插件规则,那么插件的运行环境和规则也是一个沙箱
  • 在线代码编辑器, CodeSanbox在执行脚本也会单独成立一个沙箱去隔离执行代码,防止代码访问或影响主页面
  • 微前端qiangunsingle-spa 框架里主应用和子应用之间的完全隔离,也是一种沙箱机制,如: 应用之间CSS样式不能互相影响


在了解完沙箱是什么后,那么在JavaScript语言里如何实现沙箱呢?主要有以下几种方式:

  • 使用 with 声明
  • 使用 new Function 声明
  • 基于 Proxy实现
  • 基于属性 diff实现
  • 基于 iframe实现
  • 基于 ES 提案 ShadowRealm API

前置知识


with关键字


with 扩展一个语句的作用域链。 —— MDN withJavaScript 查找某个未使用命名空间的变量时,会通过作用域链来查找,作用域链是跟执行代码的 context 或者包含这个变量的函数有关。'with'语句将某个对象添加到作用域链的顶部,如果在 statement 中有某个未使用命名空间的变量,跟作用域链中的某个属性同名,则这个变量将指向这个属性值。如果沒有同名的属性,则将拋出ReferenceError异常。

按照个人比较容易理解的意思,就是给一段代码加上指定对象为该作用的全局变量。示例代码如下:

Math.floor(1.1) // 1
// 使用with
with(Math){
    floor(1.1) // 1
}

new Function

new Function(argStr, codeStr)是能将字符串代码转换为可执行的函数。具体示例如下:

const name = 'test';
const test = new Function('arg', 'console.log(arg)');
// 这里等于 test = (arg)=> {console.log(arg)};
test(name); // test


Proxy


Proxy 对象用于创建一个对象的代理,从而实现基本操作的拦截和自定义(如属性查找、赋值、枚举、函数调用等)。Proxy MDN

Proxy只能代理object类型的变量,针对基础类型的代理只能将其封装到对象里再进行代理监听。


Proxy代理方法如下:

  • getPrototypeOf(target) 代理获取原型的方法
  • setPrototypeOf(target, newProto) 设置原型,如果不想设置原型,可以return false
  • isExtensible(target)  拦截对对象的 Object.isExtensible(),必须返回一个 Boolean 值,判断一个对象是否是可扩展的
  • preventExtensions(target) 拦截Object.preventExtensions(),让一个对象变的不可扩展,也就是永远不能再添加新的属性
  • getOwnPropertyDescriptor(target, prop) 拦截Object.getOwnPropertyDescriptor(),拦截获取对象属性的描述符
  • defineProperty(target, property, descriptor) 拦截Object.defineProperty()
  • has(target, key),针对 in 操作符的代理方法
  • get(target, property, receiver),用于拦截对象的读取属性操作
  • set(),设置属性值操作的捕获器。
  • construct(),用于拦截 new 操作符


Object.defineProperty主要区别(可拦截方法比Object.defineProperty多):

  • Proxy代理的是整个对象,Object.defineProperty只代理对象上的某个属性,如果是多层嵌套的数据需要循环递归绑定;
  • 对象上定义新属性时,Proxy可以监听到,Object.defineProperty监听不到,需要借助$set方法;
  • 数组的某些方法(push、unshift和splice)Object.defineProperty监听不到,Proxy可以监听到;

Symbol.unscopables


指用于指定对象值,其对象自身和继承的从关联对象的 with 环境绑定中排除的属性名称。

可以这么理解,就是为了防止with添加作用域的时候,将某个属性从作用域中排除掉,代码如下:

a = {
    p: 1,
    b: 2
}
// 禁止将a.p放到with作用域中
a[Symbol.unscopables] = {p: true}
with(a){
    console.log(p) // 报错 p not defined
    console.log(b) // 正常输出
}

因此很多内置对象都设置该值为true,从而降低with的侵入,具体如下:

Array.prototype[Symbol.unscopables];
/*{
  copyWithin: true,
  entries: true,
  fill: true,
  find: true,
  findIndex: true,
  flat: true,
  flatMap: true,
  includes: true,
  keys: true,
  values: true,
}*/


目录
相关文章
|
6月前
|
JavaScript Java 测试技术
基于springboot+vue.js+uniapp的企业人才引进服务平台附带文章源码部署视频讲解等
基于springboot+vue.js+uniapp的企业人才引进服务平台附带文章源码部署视频讲解等
35 0
|
7月前
|
JavaScript Java 测试技术
基于ssm+vue.js+uniapp小程序的企业人才引进服务平台附带文章和源代码部署视频讲解等
基于ssm+vue.js+uniapp小程序的企业人才引进服务平台附带文章和源代码部署视频讲解等
34 1
|
6月前
|
JSON 前端开发 JavaScript
【amis低代码前端框架】vue2集成百度低代码前端框架amis
【amis低代码前端框架】vue2集成百度低代码前端框架amis
583 0
|
Web App开发 JavaScript 前端开发
Sea.js:简单、极致的模块化Web开发体验
Sea.js 是一款现代的用于Web开发的模块加载工具,提供简单、极致的模块化体验。Sea.js 由阿里、腾讯等公司共同维护。
812 0
Sea.js:简单、极致的模块化Web开发体验
|
缓存 JavaScript 前端开发
低代码系列——js沙箱设计(二)
由于自己参与过低代码平台开发,所以希望能把我自己开发低代码中遇到的问题或者一些设计思路进行总结汇总,这是开始写的第一篇,也是比较基础的一篇,关于低代码平台的介绍会放在介绍篇章,这篇就不做过多介绍。
336 0
|
Web App开发 JavaScript 前端开发
serveless 思想 Midway.js 框架使用教程(一)
serveless 思想 Midway.js 框架使用教程(一)
455 0
serveless 思想 Midway.js 框架使用教程(一)
|
开发框架 前端开发 小程序
专注性能的多端研发框架 - ice.js 3 正式发布!
ice.js 框架在之前的版本中,主要服务于中后台 / PC 的项目研发,而随着无线端以及多端能力的拓展,ice.js 3 将成为一套面向大淘宝技术的终端应用框架。因此在 ice.js 3 的版本中除了「开发者体验」之外,还围绕「用户体验」探索了大量技术方案。 ice.js 3 地址:https://v3.ice.work/
1455 0
|
前端开发 JavaScript 算法
ivx杨帆启航React/Pixi.js/FaaS、Krpano及微服务架构
ivx杨帆启航React/Pixi.js/FaaS、Krpano及微服务架构
314 0
ivx杨帆启航React/Pixi.js/FaaS、Krpano及微服务架构
|
JSON 运维 前端开发
lowcode 低代码前端框架 amis 调研
所谓低代码开发,即无需编码或只需少量代码就可以快速生成应用程序。也就是说,企业的应用开发通过“拖拉拽”的方式即可完成。
lowcode 低代码前端框架 amis 调研
|
监控 JavaScript 前端开发
技术分享 | 测试平台开发-前端开发之Vue.js 框架的使用
技术分享 | 测试平台开发-前端开发之Vue.js 框架的使用