我正在开发一个非常简单的应用程序,以使用SSDP发现设备,并试图找到解析该命令响应的最简单方法。我试图避免做一堆NSString或正则表达式操作。
我尝试了以下两种方法:
方法1:使用GCDAsyncUdpSocket,我能够成功地发送发现命令并获得以下响应:
HTTP/1.1 200 OK
缓存控制:最大年龄=300
ST:roku:ecp
USN:uuid:roku:ecp:1234567890
扩展:
服务器:roku UPnP/1.0 MiniUPnPd/1.4
位置:http://192.168.XX.XX:8060/
这看起来像是一个常规的HTTP响应,但使用GCDAsyncUdpSocket,我将获得作为NSData对象的响应,我可以很容易地将其转换为NSString。然而,理想的做法是以某种方式将其转换为NSHTTPURLResponse,然后使用其方法来获取字段值。你知道这能不能做到吗?
方法2:我尝试使用常规的NSURLRequest来尝试发送此命令,然后我就可以得到NSHTTPURLResponse。但是,由于SSDP发现命令要求我将此请求发送到端口1900,因此我不断收到错误。我使用以下代码来发送"HTTP"请求,我知道这不是严格意义上的HTTP,但我认为这可能是一种更容易发送UDP命令的方式,因为要求看起来与HTTP非常相似。
NSURL *url = [NSURL URLWithString:@"http://239.255.255.250:1900"];
NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:url
cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:60.0];
[request setHTTPMethod:@"M-SEARCH *"];
[request setValue:@"239.255.255.250:1900" forHTTPHeaderField:@"Host"];
[request setValue:@""ssdp:discover"" forHTTPHeaderField:@"Man"];
[request setValue:@"roku:ecp" forHTTPHeaderField:@"ST"];
NSURLConnection *connection = [[NSURLConnection alloc]initWithRequest:request delegate:self startImmediately:YES];
if (connection)
{
NSLog(@"Connection success!");
}
else
{
NSLog(@"Connection failed!");
}
当我这样做时,连接是成功的,但我在NSURLConnection的didFailWithError委托中得到以下错误:
失败错误域=NSPOSIXErrorDomain代码=47"操作无法完成。协议族不支持地址族"UserInfo=0x8875940{NSErrorFailingURLKey=http://239.255.255.250:1900/,NSErrorFailingURLStringKey=http://239.255.255.250:1900/}
只有当我使用端口1900时才会发生此错误,如果我忽略了这一点,或者使用另一个更友好的HTTP端口(如8080),那么这是有效的,但很明显,我试图发现的设备不会正确响应,除非它在端口1900中收到请求。
感谢您提供的任何帮助。
端口1900对于SSDP是固定的。所有UPnP设备都固定在此端口上侦听,并且连接节点期望这样。你无法改变它。"开箱即用"是UPnP设计目标的一部分。此外,我的Cocoa专业知识非常有限,但我认为NSHTTPURLResponse
不会让事情变得更简单。[NSHTTPURLResponse allHeaderFields]
返回NSDictionary
,这意味着您不再知道头字段的原始顺序。你需要知道Ext:
之后是什么,它是一个元头。
我建议您自己解析响应,这不应该比一个周期复杂得多,用行分隔响应,然后用:
进行拆分。或者,不要尝试进行自己的SSDP握手,而是使用现成的Cocoish库,如Upnpx、Cyberlink或Platinum。仅仅在发现阶段,这可能感觉像是不合适的重炮,但我想知道在那之后,除了实际尝试调用设备上的一些操作之外,你还会对设备做什么。