我有一个由我的移动网络提供商制作的Windows桌面应用程序,可以使用SIP执行所有类型的操作:呼叫,发送消息等。此应用程序如何成功发送MESSAGE
的屏幕截图(最后4个)线):
MESSAGE
请求将作为(后面的第4行)发送:
MESSAGE sip:FROM@DOMAIN SIP/2.0
Via: SIP/2.0/UDP LOCALIP:2112;branch=z9hG4bK-d8754z-905183245f478c76-1---d8754z-;rport
Max-Forwards: 70
To: "TO"<sip:TO@DOMAIN>
From: "FROM"<sip:USERNAME@DOMAIN>;tag=63088d09
Call-ID: NGVhMDJhYzQwNmExOTQyNThmNjc5OGNmOTViNDUyYWM.
CSeq: 2 MESSAGE
Allow: INVITE, ACK, CANCEL, OPTIONS, BYE, REFER, NOTIFY, MESSAGE, SUBSCRIBE, INFO
Content-Type: text/plain
Content-Length: 4
test
并成功回应:
SIP/2.0 407 Proxy Authentication Required
Via: SIP/2.0/UDP LOCALIP:2112;received=EXTERNALIP;branch=z9hG4bK-d8754z-905183245f478c76-1---d8754z-;rport=2112
To: "TO"<sip:TO@DOMAIN>;tag=c005f0e30133ec730add76fc91f4bea
From: "FROM"<sip:USERNAME@DOMAIN>;tag=63088d09
Call-ID: NGVhMDJhYzQwNmExOTQyNThmNjc5OGNmOTViNDUyYWM.
CSeq: 2 MESSAGE
Content-Length: 0
Proxy-Authenticate: Digest nonce="3F178051B97E1F52000123000A3C53D4B",realm="DOMAIN",algorithm=MD5,qop="auth"
然后我尝试从PHP发送相同的(和n变体)请求,但我总是收到SIP/2.0 403 Forbidden
而不是SIP/2.0 407 Proxy Authentication Required
:
SIP/2.0 403 Forbidden
Via: SIP/2.0/UDP LOCALIP;received=EXTERNALIP
To: "TO"<sip:TO@DOMAIN>;tag=aprqngfrt-f7ccjj0000020
From: "FROM"<sip:USERNAME@DOMAIN>;tag=8f7be81d
Call-ID: 526576901edcc@localhost
CSeq: 1 MESSAGE
Reason: Q.850;cause=55;text="Call Terminated"
Content-Length: 0
有趣的是,如果我发送REGISTER
请求它可以正常工作,我就会成功收到SIP/2.0 401 Unauthorized
标题WWW-Authenticate
。我重新计算授权,然后重新发送。然后我收到SIP/2.0 200 OK
。它应该如何与MESSAGE
一起使用。
可能有什么不对?我错过了什么? MESSAGE
请求之前是否需要其他请求(我之前已经尝试过REGISTER
)?我上下阅读了RFC 3428,尝试了所有可能的示例,但没有成功。
答案 0 :(得分:5)
如果您查看收到的403响应,您会注意到一个Reason标题。开头的Q.850字符串表示这将是ITU-T Recomendation定义的原因代码。
具体而言,如果原因代码55与ISDN相关,文字意味着“在封闭用户组内禁止来电”(您可以在RFC 3398中查看),通常意味着在一组成员中,电话接待受到限制。
另一方面,原因55也表示请求中的问题,特别是与用户(发送者或接收者)有关的问题。下图显示了SIP用户之间正常的MESSAGE交换:
A Server B
| REGISTER | |
|--------------->| |
| 200 OK | |
|<---------------| |
| | REGISTER |
| |<--------------|
| | 200 OK |
| |-------------->|
| MESSAGE | |
|--------------->| MESSAGE |
| |-------------->|
| | 200 OK |
| |<--------------|
| 200 OK | |
|<---------------| |
实际上,严格来说,不需要来自用户A的REGISTER,但大多数系统(如IMS)将其用作身份验证机制。然后,在REGISTER请求中,特殊标头是:
Contact: <sip:USER_NAME@LOCAL_IP:LOCAL_PORT>
Expires: REGISTRATION_DURATION
请记住,200 OK回答REGISTER,可以在Expires:
标题内包含expires
标题或Contact:
参数,表示已接受到期时间。例如:
SIP/2.0 200 OK
...
Contact: <sip:USER_NAME@LOCAL_IP:LOCAL_PORT>; expires=60
...
在这种情况下,您应该在此到期时间之前重新注册(示例中为60秒)。
请记住,您正尝试向手机发送短信,接收点由您的网络提供商MGCF直接管理,因此会留下发件人的注册或MESSAGE请求。
关于您的原始MESSAGE提案,请求URI(消息的第一行)应该是:
MESSAGE sip:TO@DOMAIN SIP/2.0
因为它指的是MESSAGE接收实体。
希望这有帮助。
答案 1 :(得分:2)
正如我在评论中告诉你的那样,我不是SIP专家,但我的朋友是。我问过他你的情况,这就是他告诉我的:
SIP协议是一种对话协议,意味着每个通信都是具有唯一对话ID的对话(类似于HTTP中的会话ID)。 SIP和HTTP之间的区别在于会话ID用于不同的TPC / IP连接(HTTP请求),而对话ID在同一TPC / IP连接中使用但在不同的消息中使用。
在我看来,你在这里尝试做的有点像HTTP中的会话劫持。虽然可以在HTTP中劫持会话ID并从另一个客户端发送它,但SIP不是这样。根据我的朋友说,SIP服务器有一个内部存储器,对话ID属于哪个连接,只要知道对话ID就不能将你的信息传递给别人的对话。
你的问题并没有说明这是否是你想要做的事情,但如果是,那么我必须说你做不到。您可以发送REGISTER
命令的事实表明您与SIP服务器的通信已建立。您需要做的就是发起自己的对话并从那里开始。