我正在使用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
如果我将标头名称更改为其他名称,它会显示在请求标头中。我猜苹果在发送请求之前会替换"用户代理"字段。有办法防止这种情况发生吗?
花了两天时间研究这个问题后,我很快就找到了解决方案,创建了本机GET
和POST
方法,将它们暴露在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访问nativeGET
和nativePOST
函数。
最后一件事是发出请求并检索响应。我不知道如何在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。
现在,我们可以应用我们需要的的任何标头来进行GET
和POST
请求
post('http://example.com/', {
login: 'xxx',
password: 'yyy'
}, {
'User-Agent': 'My custom User-Agent'
})