沙箱简介
沙箱,即sandbox,顾名思义,就是让你的程序跑在一个隔离的环境下。目的是为了让不可信的代码运行在一定的环境中,从而限制这些代码访问隔离区之外的资源,不对外界的其他程序造成影响。
JS中沙箱的使用场景
前端JS中也会有应用到沙箱的时候,毕竟有时候你要获取到的是第三方的JS文件或数据,而这数据又是不一定可信的时候,创建沙箱,做好保险工作尤为重要。
- jsonp:解析服务器所返回的jsonp请求时,如果不信任jsonp中的数据,可以通过创建沙箱的方式来解析获取数据;
- 执行第三方js:当你有必要执行第三方js的时候,而这份js文件又不一定可信的时候
- 在线代码编辑器:在线代码编辑器代码的执行,基本都会放置在沙箱中,防止对页面本身造成影响;(例如:codesandbox.io/s/new)
- vue的服务端渲染:vue的服务端渲染实现中,通过创建沙箱执行前端的bundle文件;在调用createBundleRenderer方法时候,允许配置runInNewContext为true或false的形式,判断是否传入一个新创建的sandbox对象以供vm使用;
- vue模板中表达式计算:vue模板中表达式的计算被放在沙盒中,只能访问全局变量的一个白名单,如 Math 和 Date 。你不能够在模板表达式中试图访问用户定义的全局变量。 总而言之:当你要解析或执行不可信的JS的时候,当你要隔离被执行代码的执行环境的时候,当你要对执行代码中可访问对象进行限制的时候,沙箱就派上用场了。
沙箱实现一:with + new Function
首先从最简陋的方法说起,因为代码内部可以沿着作用域链往上找,进而可能篡改全局变量,所以你需要让沙箱内的变量访问都在你的监控范围内,我们可以使用with API,在with的块级作用域下,变量访问会优先查找你传入的参数对象,之后再往上找,所以相当于你变相监控到了代码中的“变量访问”:
沙箱实现二:借助iframe实现沙箱
在前端最常见的方法,还是利用iframe来构造一个沙箱。 这种方式更为方便、简单、安全,也是目前比较通用的前端实现沙箱的方案,假如你要执行的代码不是自己写的代码,不是可信的数据源,那么务必要使用iframe沙箱。sandbox是h5的提出的一个新属性, 启用方式就是在iframe标签中使用sandbox属性:
- script脚本不能执行
- 不能发送ajax请求
- 不能使用本地存储,即localStorage,cookie等
- 不能创建新的弹窗和window
- 不能发送表单
- 不能加载额外插件比如flash等 不过我们可以对这个iframe标签进行一些配置:
nodejs中的沙箱使用
nodejs中使用沙箱很简单,只需要利用原生的vm模块,便可以快速创建沙箱,同时指定上下文。
总结
即使我们知道了如何在开发过程中使用沙箱来让我们的执行环境不受影响,但是沙箱也不一定是绝对安全的,因此:
- 业务代码上不执行不可信任的第三方JS,如有必要执行第三方JS,可通过设置CSP维护白名单的方式;
- 不要信任任何用户数据源,防止恶意用户注入代码。
参考文档:
- IPC:baike.baidu.com/item/ipc/19…
- iframe标签详解:www.cnblogs.com/hzb462606/p…
- proxy:www.jianshu.com/p/c2a1aa2e2…
- with作用域:blog.csdn.net/zwkkkk1/art…
- contentWindow对象:www.cnblogs.com/goloving/p/…
- Window.postMessage:developer.mozilla.org/zh-CN/docs/…
- vm.createcontext:nodejs.cn/api/vm/vm_c…
- 沙箱:juejin.cn/post/684490…