我正在用 C# 创建一个库,以使用 Airplay 协议将照片和视频发送到我的 Apple TV(特别是与第 3 代一起使用,但希望这不重要)。
根据此规范,Airplay 的所有命令都是端口 70 上的 HTTP:http: //nto.github.com/AirPlay.html
我已经成功地在 Apple TV 上播放照片和视频,但无论我做什么,AppleTV 都只能播放 30 秒的视频。似乎我发出播放命令的 C# 客户端在 30 秒后断开连接,这导致 AppleTV 结束播放会话。
我这么认为的原因:
- 完全终止客户端应用程序会产生与等待 30 秒相同的行为(本质上是强制关闭连接)。
- 手动关闭 HttpWebRequest 或 TcpClient 连接会产生相同的行为(播放会话中途)。
- 无论我为阻止 GetResponse() 调用而保留断点多长时间,视频总是在 WebRequest 开始发送消息后 30 秒超时。
- 为视频使用不同的源(IIS、外部网络服务器)不会改变行为。
- 即使视频已经缓存在 AppleTV 上并且没有重新流式传输,超时仍然会发生。
我很确定客户端请求需要在整个视频“播放”过程中保持连接,据我所知,我已经对其进行了编码。我真的束手无策。我已经尝试了所有我能想到的方法,包括将请求作为 HttpWebRequest 和作为原始 TcpClient (两者都工作但都超时),将接收/发送超时设置为疯狂的数字,并循环读取 Tcp 流确保有“活动”。
就好像 AppleTV 期待我发送“嘿,继续播放”消息,但我还没有从网络上的任何来源看到类似的消息。我希望这只是我没有做的愚蠢的事情,因为我缺乏 Http/Tcp 知识。
这是我的代码:
Uri url = "http://somevideo.com/video.mov";
float startPosition = 0;
TcpClient tcpClient = new TcpClient("192.168.1.20",7000);
tcpClient.ReceiveTimeout = 100000;
tcpClient.SendTimeout = 100000;
//get the client stream to read data from.
NetworkStream clientStream = tcpClient.GetStream();
string body =
"Content-Location: " + url + "\n" +
"Start-Position: " + startPosition + "\n";
string request = "POST /play HTTP/1.1\n" +
"User-Agent: MediaControl/1.0\n" +
"Content-Type: text/parameters\n" +
"Content-Length: " + Encoding.ASCII.GetBytes(body).Length + "\n" +
"X-Apple-Session-ID:" + _sessionGuid.ToString() + "\n\n";
sendMessage(clientStream, request);
sendMessage(clientStream, body);
byte[] myReadBuffer = new byte[1024];
StringBuilder myCompleteMessage = new StringBuilder();
int numberOfBytesRead = 0;
//incoming message might be bigger than the buffer
do
{
try
{
numberOfBytesRead = clientStream.Read(myReadBuffer, 0, myReadBuffer.Length);
myCompleteMessage.Append(Encoding.ASCII.GetString(myReadBuffer, 0, numberOfBytesRead));
Thread.Sleep(10);//let the iOS device catch up sending data
}
catch (System.IO.IOException) { }
} while (tcpClient.Connected); //check if it's connected before checking for data available, as maybe the program might get quit and the sockets closed halfway through a read
注意:使用 telnet 我可以连接到端口 7000 上的 AppleTV 并粘贴此命令以播放整个视频:
POST /play HTTP/1.1
User-Agent: MediaControl/1.0
Content-Type: text/parameters
Content-Length: 89
X-Apple-Session-ID:fb6d816a-a5ad-4e8f-8830-9642b6e6eb35
Content-Location: http://192.168.1.11:82/2012/2012_03_11/IMG_1328.MOV
Start-Position: 0
我在端口 82 上运行 Cassini Webserver,但这也适用于 IIS。这提供了进一步的证据,表明 .Net 堆栈在 30 秒内正在做一些导致断开连接的事情。