解决 Script Error 的另类思路

2019独角兽企业重金招聘Python工程师标准>>> hot3.png

本文由小芭乐发表

前端的同学如果用 window.onerror 事件做过监控,应该知道,跨域的脚本会给出 "Script Error." 提示,拿不到具体的错误信息和堆栈信息。

这里读者可以跟我一起做一个实验,来深入了解这个事情。先做一下实验准备:

app.js

创建一个 Node APP,只做静态服务器,提供两个端口用于做跨域实验。

const express = require('express');const app = express();app.use(express.static('./public'));app.listen(3000);
app.listen(4000);

public/index.html

创建一个静态页面,监听 window.onerror 事件,并且输出事件的堆栈。同时分别加载两个域的 JS 文件。

<!DOCTYPE html>
<html>
<head><meta charset="utf-8" /><meta http-equiv="X-UA-Compatible" content="IE=edge"><title>Script Error Test</title><meta name="viewport" content="width=device-width, initial-scale=1">
</head>
<body><button id="btn-3000">3000</button><button id="btn-4000">4000</button><div><pre id="info"></pre></div>
</body>
<script>
window.addEventListener('error', evt => {const info = evt.error ? evt.error.stack : evt.message;document.querySelector('#info').textContent = info;
});
</script>
<script src="http://127.0.0.1:3000/at3000.js"></script>
<script src="http://127.0.0.1:4000/at4000.js"></script>
</html>

public/at3000.js

创建一个在 3000 端口执行的脚本,监听 3000 按钮的点击事件,并且抛出一个异常:

const btn3k = document.querySelector('#btn-3000');
btn3k.addEventListener('click', () => {throw new Error('Fail 3000');
});

public/at4000.js

同样的,创建一个在 4000 端口执行的脚本:

const btn4k = document.querySelector('#btn-4000');
btn4k.addEventListener('click', () => {throw new Error('Fail 4000');
});

复现 Script Error

这个时候,我们启动 Node APP:node app.js,然后访问 http://127.0.0.1:3000

分别点击按钮 3000 和 4000,我们发现,同域下面的 3000 按钮点击后,异常消息可以捕获到。而跨域的 4000 按钮,只有一个 Script Error。

img点击 3000 按钮

img点击 4000 按钮

我们复现了 "Script Error."!

有同学举手,我知道,只要加一个跨域头就可以了!

Access-Control-Allow-Origin

没错,我们可以给静态文件服务器加上跨域协议头:

app.use(express.static('./public', {setHeaders(res) {res.set('access-control-allow-origin', res.req.get('origin'));res.set('access-control-allow-credentials', 'true');}
}));

同时,加载 JS 的时候,加上跨域声明:

<script src="http://127.0.0.1:4000/at4000.js" crossorigin="anonymous"></script>

这样,无论 3000 还是 4000 按钮,我们点击都能获得异常信息。

但是,这个方案有两个致命的弱点:

  • 如果 JS 声明了 crossorigin="anonymous" 但是响应头没有正确,JS 会直接无法执行
  • 我们并不总是有静态服务器的配置权限,跨域头不是想加就能加

img声明了 crossorigin 但是没有响应跨域头的 JS

另类思路

如果我告诉你,可以不加跨域头,只是在 JS 文件加载之前加载一个「特别的」JS,一样可以达到目的,你信不信?

<script src="http://127.0.0.1:3000/inject-event-target.js"></script>
<script src="http://127.0.0.1:3000/at3000.js"></script>
<script src="http://127.0.0.1:4000/at4000.js"></script>

这个神奇的 inject-event-target.js 可以让我们在没有跨域头的情况下,拿到 4000 按钮事件处理器的执行异常信息。

img点击 3000

img点击 4000

如果你觉得神奇,请点赞后,继续往下阅读。这个魔法 JS,其实也很简单:

const originAddEventListener = EventTarget.prototype.addEventListener;
EventTarget.prototype.addEventListener = function (type, listener, options) {const wrappedListener = function (...args) {try {return listener.apply(this, args);}catch (err) {throw err;}}return originAddEventListener.call(this, type, wrappedListener, options);
}

原理也非笔者原创,而是从这篇文章学习而来。

简单解释一下:

  • 改写了 EventTarget 的 addEventListener 方法;
  • 对传入的 listener 进行包装,返回包装过的 listener,对其执行进行 try-catch;
  • 浏览器不会对 try-catch 起来的异常进行跨域拦截,所以 catch 到的时候,是有堆栈信息的;
  • 重新 throw 出来异常的时候,执行的是同域代码,所以 window.onerror 捕获的时候不会丢失堆栈信息;

实际上,利用包装 addEventListener,我们还可以达到「扩展堆栈」的效果:

img堆栈扩展效果

我们不仅知道异常堆栈,而且还知道导致该异常的事件处理器,是在何处添加进去的。实现这个效果,也很简单:

 (() => {const originAddEventListener = EventTarget.prototype.addEventListener;EventTarget.prototype.addEventListener = function (type, listener, options) {
+    // 捕获添加事件时的堆栈
+    const addStack = new Error(`Event (${type})`).stack;const wrappedListener = function (...args) {try {return listener.apply(this, args);}catch (err) {
+        // 异常发生时,扩展堆栈
+        err.stack += '\n' + addStack;throw err;}}return originAddEventListener.call(this, type, wrappedListener, options);}})();

同样的道理,我们也可以对 setTimeout、setInterval、requestAnimationFrame 甚至 XMLHttpRequest 做这样的拦截,得到一些我们本来得不到的信息。

此文已由作者授权腾讯云+社区发布,更多原文请点击

搜索关注公众号「云加社区」,第一时间获取技术干货,关注后回复1024 送你一份技术课程大礼包!

转载于:https://my.oschina.net/qcloudcommunity/blog/2963894

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

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

相关文章

大平台的局限

这篇文章算是二稿。初稿使的是惯用的卖弄笔法&#xff0c;写到尽兴时去查了查资料&#xff0c;哦草&#xff0c;错了好多。闷闷不乐。后来就不敢再鬼扯&#xff0c;老老实实干巴巴地讲观点。 做产品的人都喜欢大平台&#xff0c;好像男人都喜欢大胸脯女郎&#xff0c;但是胸脯大…

Lisenter笔记

EventListener与EventObject要完成在线用户列表的监听器&#xff0c;需要使用如下几个接口&#xff1a;ServletContextListener接口&#xff1a;在上下文初始化时设置一个空的集合到application之中&#xff1b;HttpSessionAttributeListener接口&#xff1a;用户增加session属…

Android应用开发—重载fragment构造函数导致的lint errors

背景&#xff1a;在一次release打包中发现lint报以下错误&#xff1a; Error: Avoid non-default constructors in fragments: use a default constructor plus Fragment#setArguments(Bundle) instead [ValidFragment] 根据后面的log提示是由于重载了fragment的构造函数&…

迅雷影音怎样 1.5倍速度播放

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到教程。 看视频 觉得播放速度太慢&#xff0c;想让1.5速度播放可以这样设置&#xff1a; 点击快进按钮&#xff0c;点一次变为1.1倍&#xff0c…

【Java】Mybatis mapper动态代理方式

前言 我们在使用Mybatis的时候&#xff0c;获取需要执行的SQL语句的时候&#xff0c;都是通过调用xml文件来获取&#xff0c;例如&#xff1a;User user (User) sqlSession.selectOne("cn.ddnd.www.Entity.User.getUser", "xue8qq.com");。这种方式是通过…

git pull时冲突的几种解决方式

仅结合本人使用场景&#xff0c;方法可能不是最优的 1. 忽略本地修改&#xff0c;强制拉取远程到本地 主要是项目中的文档目录&#xff0c;看的时候可能多了些标注&#xff0c;现在远程文档更新&#xff0c;本地的版本已无用&#xff0c;可以强拉 git fetch --allgit reset --h…

Android应用开发—eventBus发布事件和事件处理的时序关系

占坑&#xff0c;简单说明下eventBus发布事件和事件处理的时序关系。 什么时候使用sticky&#xff1a; 当你希望你的事件不被马上处理的时候&#xff0c;举个栗子&#xff0c;比如说&#xff0c;在一个详情页点赞之后&#xff0c;产生一个VoteEvent&#xff0c;VoteEvent并不立…

grep命令 解说

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到教程。 grep&#xff08;global search regular expression(RE) and print out the line&#xff0c;全面搜索正则表达式并把行打印出来&#x…

创业第一桶金怎么来

文章摘要&#xff1a;资金是创业要具备的一个必要条件&#xff0c;那么对于创业者来说&#xff0c;第一桶金如何取得&#xff1f;资金是创业要具备的一个必要条件&#xff0c;那么对于创业者来说&#xff0c;第一桶金如何取得&#xff1f;   一、一门手艺   都说拥有万贯…

4001.基于双向链表的双向冒泡排序法

基于双向链表的双向冒泡排序法 发布时间: 2018年11月26日 10:09 时间限制: 1000ms 内存限制: 128M 习题集源码中出现了 temp->next->prior p; 本人推断这里缺少预先的对temp->nextNULL这种情况的判定&#xff0c;所以需加入一个判断语句解决。 此为非循环的双向链…

页面向上滚动

#页面或者div向上无缝滚动 1.css: body {margin: 0;padding: 0;overflow: hidden;}.container {position: relative;top: 0;}.container div {width: 500px;height: 500px;border: 1px solid chartreuse;font-size: 100px;line-height: 500px;font-weight: bold;color: black;t…

叨逼叨

此处记录点零散的小idea&#xff0c;为了避免把csdn当微博&#xff0c;开一篇&#xff0c;都记在这里吧。 感觉服务注册机制&#xff0c;貌似也是一种依赖注入。&#xff08;虽然我还没完全搞懂依赖注入&#xff09;&#xff0c;理由呢&#xff1a;你需要一个模块的功能&#x…

Linux:echo命令详解

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到教程。 echo命令 用于字符串的输出 格式 echo string使用echo实现更复杂的输出格式控制 1.显示普通字符串: echo "It is a test"这里…

看年轻人如何赚第一桶金

上世纪90年代&#xff0c;成为百万富翁&#xff0c;对很多人只是个梦想。不过如今&#xff0c;随着经济飞速发展&#xff0c;拥有百万资产已经不再是神话&#xff0c;放眼望去&#xff0c;我们身边的百万富翁比比皆是&#xff0c;甚至很多初入社会、白手起家的年轻人&#xff0…

跨越解决方案之nginx

这里是修真院前端小课堂&#xff0c;每篇分享文从 【背景介绍】【知识剖析】【常见问题】【解决方案】【编码实战】【扩展思考】【更多讨论】【参考文献】 八个方面深度解析前端知识/技能&#xff0c;本篇分享的是&#xff1a; 【跨越解决方案之nginx】 1.背景介绍 跨域&#x…

学习 shell脚本之前的基础知识

见 : http://www.92csz.com/study/linux/12.htm【什么是shell】 简单点理解&#xff0c;就是系统跟计算机硬件交互时使用的中间介质&#xff0c;它只是系统的一个工具。实际上&#xff0c;在shell和计算机硬件之间还有一层东西那就是系统内核了。打个比方&#xff0c;如果把计算…

「分块系列」数列分块入门3 解题报告

数列分块入门3 题意概括 区间加法&#xff0c;区间求前驱。 写在前面 这题的方法与分块2方法极其类似&#xff0c;建议自行解决。 正题 和上一题类似&#xff0c;但是二分不是用来计数的&#xff0c;而是用来求小于c的最大值的。然后对于不完整快&#xff0c;将小于c的值求最大…

创业者自述:我的第一桶金是如何来的

记者采访王宏筠的当天&#xff0c;北京气温已达到30℃&#xff0c;王宏筠从他的铁灰色奥迪A6车上下来&#xff0c;一身挺括的西装&#xff0c;打着领带&#xff0c;肩上背着一个超大的牛皮包。后来他对记者说&#xff0c;穿西服是因为多年在外企养成的习惯&#xff0c;一年中至…

Git cherry-pick后再merge出现一个“奇怪”的现象

背景描述&#xff1a;有的时候基于一个master branch拉出一个独立feature分支做开发时&#xff0c;两条分支都在并行开发&#xff0c;如果master分支增加了某些功能&#xff0c;解决了某些关键bug&#xff0c;而独立feature分支不需要所有的增加的commit&#xff0c;只需要某一…

inux系统中如何进入退出vim编辑器

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到教程。 VIM编辑器&#xff0c;可以新建文件也可以修改文件&#xff0c;命令为&#xff1a;vim AAA 。AAA就是文件名。 如果这个文件&#xff…