WKWebview在NSURLProtocol中body丢失问题的思考

背景说明

之所以想hook XMLHTTPRequest是因为NSURLProtocol导流WKWebview有body丢失问题。要解决这个问题,目前一个思路,通过注入JS代码的方式来解决这个问题,在NSURLProtocol的didReceiveData函数中,我们可以第一时间接收到请求返回的HTML数据,然后我们把hook XMLHTTPRequest的send方法和open的JS函数插入到HTML之中,然后再返回给系统。

在hook代码里面要重组url和body参数并且加上标识位,然后重新发起get请求,然后这时候我们的NSURLProtocol会再次捕获这个重新发起的get请求,根据我们请求中的标记,辨别出是我们重组过的请求,将请求中的参数取出来(我们把body存放在文件里,做好标识,便于一一对应request,方便读取),再次拼接以后塞入body发起post请求。

1.Hook XMLHTTPRequest

我打算分成两步来进行,第一部分已经完成,第二部分还在进行中。

1.1 hook XMLHTTPRequest发起的请求

XMLHTTPRequest的hook,参考https://github.com/wendux/Ajax-hook 使用起来很简单,把hookAjax函数插入script标签就可以了。如下范例,

<script src="https://unpkg.com/ajax-hook/dist/ajaxhook.min.js"></script>///这行插入到HTML文件的head标签内
///如下内容插入到body标签内
<script>

    hookAjax(
        // hook functions and callbacks of XMLHttpRequest object
        {
            onreadystatechange: function (xhr) {
                console.log("onreadystatechange called: %O", xhr)
                //return true
            },
            onload: function (xhr) {
                console.log("onload called: %O", xhr)
                xhr.responseText = "hook" + xhr.responseText;
                //return true;
            },
            open: function (arg, xhr) {
                console.log("open called: method:%s,url:%s,async:%s", arg[0], arg[1], arg[2], xhr)
                arg[1] += "?hook_tag=1";
                //统一添加请求头

            },
            send: function (arg, xhr) {
                console.log("send called: %O", arg[0])
                xhr.setRequestHeader("_custom_header_", "ajaxhook")
            },
            setRequestHeader: function (arg, xhr) {
                console.log("setRequestHeader called!", arg)
            },

            // hook attributes of XMLHttpRequest object
            timeout: {
                setter: function (v, xhr) {
                    //timeout shouldn't exceed 10s
                    return Math.max(v, 1000);
                }
            }
        }
    );

    $.get().done(function (d) {
        console.log(d.substr(0, 30) + "...")
        //use original XMLHttpRequest
        console.log("unhook")
        unHookAjax()
        $.get().done(function (d) {
            console.log(d.substr(0, 10))
        })

    })

</script>

通过控制台的log,验证后,可以拦截到请求。

1.2 将hook代码接入NSURLProtocol

时间关系,这部分还在验证。这是下个阶段的任务。

2.另一种思路

XMLHTTPRequest是否有设置proxy的方式?如果可行的话,是不是就一步登天了?可惜,并不能,不过,有个老哥提出了另一种方式,https://juejin.im/post/5db85842f265da4d1f51bbda ,这种方式也算是hook的变种。
这种方式待验证.

update - 2023/7/26
时隔三年,ChatGPT横空出世,重新挖出这个问题,看看ChatGPT给的解决思路
当需要在WKWebView中获取带有body的HTTP请求的body数据时,可以通过以下切实可行的方案实现JS和原生代码的交互:

  1. 在JS中获取请求的body数据
    在页面的JS中,可以通过XMLHttpRequestfetch等方式发起HTTP请求,并将请求的body数据提取出来,然后将它传递给原生代码。为了实现这一点,你可以在JS中监听页面的网络请求,获取请求的body数据,并通过window.webkit.messageHandlers与原生代码交互。

    例如,在JS中,可以这样获取请求的body数据:

    // 发起XMLHttpRequest请求
    var xhr = new XMLHttpRequest();
    xhr.open("POST", "https://example.com/api", true);
    
    // 设置请求头,根据需要进行调整
    xhr.setRequestHeader("Content-Type", "application/json");
    
    // 设置请求完成的回调
    xhr.onreadystatechange = function () {
      if (xhr.readyState === XMLHttpRequest.DONE) {
        // 请求完成,将body数据传递给原生代码
        var requestBody = xhr.responseText; // 这里假设请求返回的是JSON数据,如果是其他格式,需要相应调整
        window.webkit.messageHandlers.getData.postMessage(requestBody);
      }
    };
    
    // 发送请求,假设请求体为JSON格式
    var requestBodyJSON = { "key": "value" };
    var requestBodyString = JSON.stringify(requestBodyJSON);
    xhr.send(requestBodyString);
    
  2. 原生代码处理JS传递的body数据
    在原生代码中,你需要为WKWebView设置一个JavaScript交互的接口,以便接收来自JS的数据。这可以通过WKWebViewWKScriptMessageHandler实现。

    首先,创建一个遵循WKScriptMessageHandler协议的自定义类,用于处理JS传递的数据:

    import UIKit
    import WebKit
    
    class MessageHandler: NSObject, WKScriptMessageHandler {
        func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) {
            if let requestBody = message.body as? String {
                // 在这里可以获取到JS传递的body数据
                print("请求的body数据:", requestBody)
            }
        }
    }
    
  3. 将自定义MessageHandler与WKWebView关联
    在原生代码中,将自定义的MessageHandlerWKWebView关联,并添加需要交互的JavaScript脚本:

    import UIKit
    import WebKit
    
    class ViewController: UIViewController, WKNavigationDelegate {
        var webView: WKWebView!
        
        override func viewDidLoad() {
            super.viewDidLoad()
            
            // 创建WKWebView配置
            let configuration = WKWebViewConfiguration()
            
            // 创建并关联MessageHandler
            let messageHandler = MessageHandler()
            let userContentController = WKUserContentController()
            userContentController.add(messageHandler, name: "getData") // 这里的名称要与JS中postMessage的名称一致
            configuration.userContentController = userContentController
            
            // 创建WKWebView并加载页面
            webView = WKWebView(frame: view.bounds, configuration: configuration)
            webView.navigationDelegate = self
            view.addSubview(webView)
            
            if let url = Bundle.main.url(forResource: "index", withExtension: "html") {
                webView.loadFileURL(url, allowingReadAccessTo: url)
            }
        }
    }
    
  4. 在JS中调用原生代码的接口
    在JS中,通过window.webkit.messageHandlers.getData.postMessage调用原生代码处理请求的body数据。

  5. 在原生代码中处理获取到的body数据
    在原生代码的MessageHandler中,你可以将获取到的body数据进行进一步处理,或者传递给其他需要的地方。

通过以上步骤,你可以在WKWebView中实现JS和原生代码的交互,从而获取到带有body的HTTP请求的body数据。请注意,由于涉及到与网页JS的交互,需要确保JS代码的正确性和安全性。另外,由于涉及到JS和原生代码的交互,如果你需要在多个地方使用这个功能,请考虑代码的复用性和整体的架构设计。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,137评论 6 511
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,824评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,465评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,131评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,140评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,895评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,535评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,435评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,952评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,081评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,210评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,896评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,552评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,089评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,198评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,531评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,209评论 2 357