如何通过SIP发送即时消息

时间:2013-10-21 18:57:13

标签: php sip

我有一个由我的移动网络提供商制作的Windows桌面应用程序,可以使用SIP执行所有类型的操作:呼叫,发送消息等。此应用程序如何成功发送MESSAGE的屏幕截图(最后4个)线): Wireshark

来自桌面应用程序的

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,尝试了所有可能的示例,但没有成功。

2 个答案:

答案 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服务器的通信已建立。您需要做的就是发起自己的对话并从那里开始。