qb-pc-sdk 1.0.5 → 1.0.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/index.js CHANGED
@@ -14,6 +14,11 @@ if (isBrowser && typeof window !== 'undefined' && window.console && !window.cons
14
14
  // 过滤函数,检查是否应该过滤
15
15
  const shouldFilter = function(...args) {
16
16
  const errorText = args.join(' ').toLowerCase();
17
+ // 检查是否包含 localhost.weixin.qq.com(各种格式)
18
+ const hasWeixinLocalhost = /https?:\/\/localhost\.weixin\.qq\.com/i.test(errorText) ||
19
+ /localhost\.weixin\.qq\.com/i.test(errorText) ||
20
+ errorText.includes('localhost.weixin.qq.com');
21
+
17
22
  return (
18
23
  errorText.includes('微信') ||
19
24
  errorText.includes('wechat') ||
@@ -22,7 +27,7 @@ if (isBrowser && typeof window !== 'undefined' && window.console && !window.cons
22
27
  errorText.includes('微信游戏') ||
23
28
  errorText.includes('微信小游戏') ||
24
29
  errorText.includes('wxgamesdkframe') ||
25
- errorText.includes('localhost.weixin.qq.com') ||
30
+ hasWeixinLocalhost ||
26
31
  errorText.includes('wx_game_base') ||
27
32
  errorText.includes('getgamesession') ||
28
33
  errorText.includes('xmlhttprequest') ||
@@ -31,7 +36,14 @@ if (isBrowser && typeof window !== 'undefined' && window.console && !window.cons
31
36
  errorText.includes('cors policy') ||
32
37
  errorText.includes('secure context') ||
33
38
  errorText.includes('loopback') ||
34
- errorText.includes('more-private address space')
39
+ errorText.includes('more-private address space') ||
40
+ errorText.includes('net::') ||
41
+ errorText.includes('timeout') ||
42
+ errorText.includes('frame on invoke') ||
43
+ errorText.includes('/wx_game_base/api/business') ||
44
+ /localhost\.weixin\.qq\.com:\d+/.test(errorText) ||
45
+ /:\d{5}\/wx_game_base/.test(errorText) ||
46
+ /:\d{5}\/wx_game_base/.test(errorText)
35
47
  );
36
48
  };
37
49
 
@@ -87,10 +99,101 @@ if (isBrowser && typeof window !== 'undefined' && window.console && !window.cons
87
99
  }
88
100
  });
89
101
 
102
+ // 拦截 XMLHttpRequest 的错误事件,只隐藏错误日志,不阻止请求
103
+ if (window.XMLHttpRequest) {
104
+ const OriginalXHR = window.XMLHttpRequest;
105
+ window.XMLHttpRequest = function() {
106
+ const xhr = new OriginalXHR();
107
+ const originalAddEventListener = xhr.addEventListener;
108
+
109
+ // 拦截 addEventListener,过滤错误事件
110
+ xhr.addEventListener = function(type, listener, options) {
111
+ if (type === 'error' || type === 'loadend' || type === 'abort') {
112
+ const wrappedListener = function(event) {
113
+ const url = (xhr.responseURL || xhr._requestURL || '').toLowerCase();
114
+ // 如果请求被标记为需要过滤,或者 URL 匹配过滤规则,则静默处理
115
+ if (xhr._qbsdkFiltered || shouldFilter(url)) {
116
+ return; // 静默处理错误事件
117
+ }
118
+ if (listener) {
119
+ listener.apply(this, arguments);
120
+ }
121
+ };
122
+ return originalAddEventListener.call(this, type, wrappedListener, options);
123
+ }
124
+ return originalAddEventListener.apply(this, arguments);
125
+ };
126
+
127
+ // 保存请求 URL 以便后续判断
128
+ const originalOpen = xhr.open;
129
+ xhr.open = function(method, url, async, user, password) {
130
+ xhr._requestURL = url;
131
+ // 如果是 localhost.weixin.qq.com 的请求,标记为需要过滤
132
+ if (url && /localhost\.weixin\.qq\.com/i.test(url)) {
133
+ xhr._qbsdkFiltered = true;
134
+ }
135
+ return originalOpen.apply(this, arguments);
136
+ };
137
+
138
+ // 拦截 onerror 属性
139
+ Object.defineProperty(xhr, 'onerror', {
140
+ set: function(handler) {
141
+ const wrappedHandler = function(event) {
142
+ const url = (xhr.responseURL || xhr._requestURL || '').toLowerCase();
143
+ // 如果请求被标记为需要过滤,或者 URL 匹配过滤规则,则静默处理
144
+ if (!xhr._qbsdkFiltered && !shouldFilter(url) && handler) {
145
+ handler.apply(this, arguments);
146
+ }
147
+ };
148
+ originalAddEventListener.call(this, 'error', wrappedHandler);
149
+ },
150
+ get: function() {
151
+ return this._onerror;
152
+ },
153
+ configurable: true
154
+ });
155
+
156
+ return xhr;
157
+ };
158
+ }
159
+
90
160
  window._qbsdkErrorHandlerAdded = true;
91
161
  }
92
162
  }
93
163
 
164
+ // 在加载底层SDK之前,拦截并阻止所有对 localhost.weixin.qq.com 的请求
165
+ if (isBrowser && window.XMLHttpRequest && !window._qbsdkXHRBlocked) {
166
+ const OriginalXHR = window.XMLHttpRequest;
167
+ window.XMLHttpRequest = function() {
168
+ const xhr = new OriginalXHR();
169
+ const originalOpen = xhr.open;
170
+ const originalSend = xhr.send;
171
+
172
+ // 拦截 open 方法,阻止对 localhost.weixin.qq.com 的请求
173
+ xhr.open = function(method, url, async, user, password) {
174
+ // 如果是微信游戏SDK的请求,直接阻止
175
+ if (url && /localhost\.weixin\.qq\.com/i.test(url)) {
176
+ xhr._qbsdkBlocked = true;
177
+ // 不调用原始 open,直接返回,阻止请求
178
+ return;
179
+ }
180
+ return originalOpen.apply(this, arguments);
181
+ };
182
+
183
+ // 拦截 send 方法,如果请求被阻止则不发送
184
+ xhr.send = function(data) {
185
+ if (xhr._qbsdkBlocked) {
186
+ // 请求已被阻止,不发送
187
+ return;
188
+ }
189
+ return originalSend.apply(this, arguments);
190
+ };
191
+
192
+ return xhr;
193
+ };
194
+ window._qbsdkXHRBlocked = true;
195
+ }
196
+
94
197
  // 浏览器中预加载底层 SDK(仅 side-effect,挂载 window.GDTAdSDK)
95
198
  if (isBrowser) {
96
199
  try {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "qb-pc-sdk",
3
- "version": "1.0.5",
3
+ "version": "1.0.7",
4
4
  "description": "趣变广告 SDK 封装器 - PC端广告接入工具,自动处理多级接口映射与原生渲染逻辑",
5
5
  "main": "index.js",
6
6
  "files": [
@@ -7,91 +7,136 @@
7
7
 
8
8
  // 兼容浏览器和 Node.js 环境
9
9
  const window = typeof global !== 'undefined' && global.window ? global.window : (typeof window !== 'undefined' ? window : global);
10
+ const isBrowser = typeof window !== 'undefined' && typeof window.document !== 'undefined';
10
11
 
11
- // 拦截并过滤微信游戏SDK相关的错误日志(仅拦截一次)
12
- if (typeof window !== 'undefined' && window.console && !window.console.error._qbsdkFiltered) {
13
- const originalError = window.console.error;
14
- const originalWarn = window.console.warn;
15
-
16
- // 过滤函数,检查是否应该过滤
17
- const shouldFilter = function(...args) {
18
- const errorText = args.join(' ').toLowerCase();
19
- return (
20
- errorText.includes('微信') ||
21
- errorText.includes('wechat') ||
22
- errorText.includes('weixin') ||
23
- errorText.includes('wxgame') ||
24
- errorText.includes('微信游戏') ||
25
- errorText.includes('微信小游戏') ||
26
- errorText.includes('wxgamesdkframe') ||
27
- errorText.includes('localhost.weixin.qq.com') ||
28
- errorText.includes('wx_game_base') ||
29
- errorText.includes('getgamesession') ||
30
- errorText.includes('xmlhttprequest') ||
31
- errorText.includes('err_failed') ||
32
- errorText.includes('err_connection_refused') ||
33
- errorText.includes('cors policy') ||
34
- errorText.includes('secure context') ||
35
- errorText.includes('loopback') ||
36
- errorText.includes('more-private address space')
37
- );
38
- };
39
-
40
- const filteredError = function(...args) {
41
- // 如果不是微信相关的错误,正常输出
42
- if (!shouldFilter(...args)) {
43
- originalError.apply(window.console, args);
44
- }
45
- };
46
-
47
- const filteredWarn = function(...args) {
48
- // 如果不是微信相关的警告,正常输出
49
- if (!shouldFilter(...args)) {
50
- originalWarn.apply(window.console, args);
12
+ // ============================================================
13
+ // 核心拦截逻辑:在 SDK 加载前彻底屏蔽 WeChat Localhost 请求
14
+ // ============================================================
15
+ if (isBrowser) {
16
+ // 1. 拦截 XMLHttpRequest
17
+ if (window.XMLHttpRequest && !window._qbsdkXHRBlocked) {
18
+ const OriginalXHR = window.XMLHttpRequest;
19
+
20
+ // 定义一个安全的 XHR 代理类
21
+ window.XMLHttpRequest = function() {
22
+ const xhr = new OriginalXHR();
23
+ const originalOpen = xhr.open;
24
+ const originalSend = xhr.send;
25
+
26
+ // 标记此实例是否被阻断
27
+ xhr._qbsdkBlocked = false;
28
+
29
+ xhr.open = function(method, url, async, user, password) {
30
+ const targetUrl = (url || '').toString().toLowerCase();
31
+
32
+ // 严格匹配 localhost.weixin.qq.com 及其变体
33
+ if (targetUrl.includes('localhost.weixin.qq.com') ||
34
+ targetUrl.includes('wx_game_base') ||
35
+ /127\.0\.0\.1:\d+\/wx_game_base/.test(targetUrl)) {
36
+
37
+ xhr._qbsdkBlocked = true;
38
+
39
+ // 伪造一个成功的状态,防止 SDK 内部抛出未捕获异常导致重试
40
+ // 我们不调用 originalOpen,而是让它"空转"
41
+ return;
42
+ }
43
+
44
+ return originalOpen.apply(this, arguments);
45
+ };
46
+
47
+ xhr.send = function(data) {
48
+ if (xhr._qbsdkBlocked) {
49
+ // 模拟异步响应,防止同步阻塞
50
+ setTimeout(() => {
51
+ // 如果 SDK 监听了 error 或 loadend,我们需要伪造一个响应
52
+ // 但为了静默,通常什么都不做最好,或者触发一个假的 load 事件
53
+ // 这里选择静默失败,不触发网络层面的报错
54
+ if (xhr.readyState !== 4) {
55
+ Object.defineProperty(xhr, 'readyState', { value: 4, writable: true });
56
+ Object.defineProperty(xhr, 'status', { value: 0, writable: true }); // 0 status usually implies network error but internally handled
57
+ Object.defineProperty(xhr, 'statusText', { value: 'Blocked by AdSDKWrapper', writable: true });
58
+ }
59
+
60
+ // 触发 fake error 事件以满足某些 SDK 的超时逻辑,但不会在 Console 报 net::ERR
61
+ if (xhr.onerror) xhr.onerror(new ProgressEvent('error'));
62
+ }, 0);
63
+ return;
64
+ }
65
+ return originalSend.apply(this, arguments);
66
+ };
67
+
68
+ return xhr;
69
+ };
70
+
71
+ // 复制静态属性 (如 UNSENT, OPENED 等)
72
+ for (let key in OriginalXHR) {
73
+ if (Object.prototype.hasOwnProperty.call(OriginalXHR, key)) {
74
+ window.XMLHttpRequest[key] = OriginalXHR[key];
75
+ }
51
76
  }
52
- };
53
-
54
- // 标记已过滤,避免重复拦截
55
- filteredError._qbsdkFiltered = true;
56
- filteredWarn._qbsdkFiltered = true;
57
- window.console.error = filteredError;
58
- window.console.warn = filteredWarn;
59
-
60
- // 拦截全局错误事件,过滤微信游戏SDK相关的错误
61
- if (!window._qbsdkErrorHandlerAdded) {
62
- const originalOnError = window.onerror;
63
- window.onerror = function(message, source, lineno, colno, error) {
64
- const errorText = (message || '').toLowerCase() + (source || '').toLowerCase();
65
- if (shouldFilter(errorText)) {
66
- // 静默处理,不输出
67
- return true; // 阻止默认错误处理
77
+
78
+ window._qbsdkXHRBlocked = true;
79
+ }
80
+
81
+ // 2. 拦截 Fetch API (很多现代 SDK 混用 XHR 和 Fetch)
82
+ if (window.fetch && !window._qbsdkFetchBlocked) {
83
+ const originalFetch = window.fetch;
84
+ window.fetch = function(input, init) {
85
+ let url = '';
86
+ if (typeof input === 'string') {
87
+ url = input;
88
+ } else if (input instanceof Request) {
89
+ url = input.url;
68
90
  }
69
- // 其他错误正常处理
70
- if (originalOnError) {
71
- return originalOnError.call(this, message, source, lineno, colno, error);
91
+
92
+ url = url.toLowerCase();
93
+
94
+ if (url.includes('localhost.weixin.qq.com') || url.includes('wx_game_base')) {
95
+ // 返回一个永远 pending 或者 立即 reject 的 Promise
96
+ // 为了不报错,返回一个伪造的 Response
97
+ return Promise.resolve(new Response(JSON.stringify({ code: -1, msg: 'Blocked' }), {
98
+ status: 200,
99
+ statusText: 'OK',
100
+ headers: { 'Content-Type': 'application/json' }
101
+ }));
72
102
  }
73
- return false;
103
+
104
+ return originalFetch.apply(this, arguments);
74
105
  };
106
+ window._qbsdkFetchBlocked = true;
107
+ }
108
+
109
+ // 3. 拦截 Console (保留原有的日志过滤逻辑作为第二道防线)
110
+ if (window.console && !window.console.error._qbsdkFiltered) {
111
+ const originalError = window.console.error;
112
+ const originalWarn = window.console.warn;
75
113
 
76
- // 拦截未捕获的 Promise rejection
77
- const originalUnhandledRejection = window.onunhandledrejection;
78
- window.addEventListener('unhandledrejection', function(event) {
79
- const reason = event.reason || {};
80
- const errorText = (reason.message || reason || '').toString().toLowerCase();
81
- if (shouldFilter(errorText)) {
82
- // 阻止默认行为,静默处理
83
- event.preventDefault();
84
- return;
85
- }
86
- // 其他错误正常处理
87
- if (originalUnhandledRejection) {
88
- originalUnhandledRejection.call(window, event);
89
- }
90
- });
114
+ const shouldFilter = (args) => {
115
+ const text = args.map(String).join(' ').toLowerCase();
116
+ return (
117
+ text.includes('localhost.weixin.qq.com') ||
118
+ text.includes('wx_game_base') ||
119
+ text.includes('err_connection_refused') ||
120
+ text.includes('err_failed') ||
121
+ text.includes('cors policy') ||
122
+ text.includes('wxgamesdkframe')
123
+ );
124
+ };
125
+
126
+ window.console.error = function(...args) {
127
+ if (!shouldFilter(args)) originalError.apply(console, args);
128
+ };
91
129
 
92
- window._qbsdkErrorHandlerAdded = true;
130
+ window.console.warn = function(...args) {
131
+ if (!shouldFilter(args)) originalWarn.apply(console, args);
132
+ };
133
+
134
+ window.console.error._qbsdkFiltered = true;
93
135
  }
94
136
  }
137
+ // ============================================================
138
+ // 拦截逻辑结束
139
+ // ============================================================
95
140
 
96
141
  // 内部常量配置
97
142
  const API_CONFIG = {
@@ -316,6 +361,11 @@
316
361
  }
317
362
 
318
363
  _handleError(error, message) {
364
+ // 内部错误也进行过滤,防止将过滤掉的错误再次打印
365
+ const errStr = String(error).toLowerCase();
366
+ if (errStr.includes('localhost.weixin.qq.com') || errStr.includes('blocked by adsdkwrapper')) {
367
+ return;
368
+ }
319
369
  console.error(`[AdSDK Error] ${message}:`, error);
320
370
  if (this.config.onAdError) {
321
371
  this.config.onAdError(error, message);
@@ -344,4 +394,4 @@
344
394
 
345
395
  // 返回类供直接使用
346
396
  return AdSDKWrapper;
347
- })(typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : this);
397
+ })(typeof window !== 'undefined' ? window : typeof global !== 'undefined' ? global : this);