更改TVML应用程序中XMLHttpRequest的用户代理



我正在使用TVMLKit开发苹果电视应用程序。我的应用程序的JavaScript代码尝试使用XMLHttpRequest向服务器发送HTTP请求。服务器需要一个特定的用户代理,所以我尝试了这个:

var request = new XMLHttpRequest();
request.open("GET", url, true);
request.setRequestHeader("User-Agent", "MyApp");
request.send();

服务器接收不同的用户代理标头:

User-Agent: <Projectname>/1 CFNetwork/758.1.6 Darwin/15.0.0

如果我将标头名称更改为其他名称,它会显示在请求标头中。我猜苹果在发送请求之前会替换"用户代理"字段。有办法防止这种情况发生吗?

花了两天时间研究这个问题后,我很快就找到了解决方案,创建了本机GETPOST方法,将它们暴露在javascript中。这不是最好的解决方案,但我仍然想分享它。也许它可以帮助别人。

以下是的工作原理

首先,我们需要安装Alamofire库。我们将使用它来创建请求。

github上的Readme有安装它所需的所有说明

安装Alamofire后,我们需要将其导入AppDelegate.swift

import Alamofire

然后,我们需要在应用程序控制器(AppDelegate.swift)中创建函数,该函数将向javascript 公开方法

func appController(appController: TVApplicationController, evaluateAppJavaScriptInContext jsContext: JSContext)
{
    let requests = [String : AnyObject]()
    let get: @convention(block) (String, String, [String : String]?) -> Void = { (cId:String, url:String, headers:[String : String]?) in
        Alamofire.request(.GET, url, headers: headers)
            .responseString { response in
                jsContext.evaluateScript("requests." + cId + "(" + response.result.value! + ")")
        }
    }
    let post: @convention(block) (String, String, [String : AnyObject]?, [String : String]?) -> Void = { (cId:String, url:String, parameters:[String : AnyObject]?, headers:[String : String]?) in
        Alamofire.request(.POST, url, parameters: parameters, headers: headers)
            .responseString { response in
                jsContext.evaluateScript("requests." + cId + "(" + response.result.value! + ")")
        }
    }
    jsContext.setObject(requests, forKeyedSubscript: "requests");
    jsContext.setObject(unsafeBitCast(get, AnyObject.self), forKeyedSubscript: "nativeGET");
    jsContext.setObject(unsafeBitCast(post, AnyObject.self), forKeyedSubscript: "nativePOST");
}

AppDelegate.swift的完整代码,你可以在这里找到

一切就绪!现在,我们可以从javascript访问nativeGETnativePOST函数。

最后一件事是发出请求并检索响应。我不知道如何在swift中执行回调,所以我使用了jsonp方法,使用运行时生成的函数并将它们的名称传递给本机函数。

以下是它在javascript 中的外观

export function get(url, headers = {}) {
    return new Promise((resolve) => {
        const cId = `get${Date.now()}`;
        requests[cId] = response => {
            delete requests[cId];
            resolve(response);
        }
        nativeGET(cId, url, headers);
    });
}
export function post(url, parameters = {}, headers = {}) {
    return new Promise((resolve) => {
        const cId = `post${Date.now()}`;
        requests[cId] = response => {
            delete requests[cId];
            resolve(response);
        }
        nativePOST(cId, url, parameters, headers);
    });
}

上面的代码是用ES6编写的,您需要在TVJS应用程序中包含Promise polifill。

现在,我们可以应用我们需要的的任何标头来进行GETPOST请求

post('http://example.com/', {
    login: 'xxx', 
    password: 'yyy'
}, {
    'User-Agent': 'My custom User-Agent'
})

相关内容

  • 没有找到相关文章

最新更新