SIP的路由机制是SIP协议的灵魂,通过此路由机制,才使得一个呼叫请求在VOIP网络的众多Proxy节点中,找到正确的目的端。在了解SIP路由机制前,我们先弄清以下几个感念。
1 与路由的相关概念
-
请求路由(Route)与响应路由(via)
按场景来分可以把SIP路由分为请求路由与响应路由两种路由。
请求路由(Route):发起端到目的端的路径。比如发起端发起INVITE呼叫请求消息所走过的路径或者BYE结束一个呼叫的请求消息所走过的路径就为请求路由。
请求路由的生成是通过头域Record-Routes进行收集,详见下面的Record-Route与Route介绍。
响应路由(via):对请求响应的消息所走的路径,比如目的端对INVITE的响应消息 RING 或 200OK所走的路径。
响应路由的生成是通过via头域进行收集。一个SIP消息每经过一个Proxy(包括主叫),都会被加上一个Via头域,当消息到达被叫后,Via头域就记录了请求消息经过的完整路径。被叫将这些Via头域原样copy到响应消息中(包括各Via的参数,以及各Via的顺序),然后下发给第一个Via中的URI,每个Proxy转发响应消息前都会把第一个Via(也就是它自己添加的Via)删除,然后将消息转发给新的第一个Via中的URI,直到消息到达主叫。
-
路由收集(Record-Route)与路由(Route)
路由收集(Record-Route):请求消息被代理转发前,由代理服务器插入请求消息中,这样可以使该对话中将来的请求仍能经过该Proxy。
路由(Route):发起端收到响应确认消息后,根据响应消息中的Record-Route头域获得一个Proxy列表,用来指示后续请求消息的路由;同样,目的端收到请求消息后,根据请求消息的Record-Route头域获得一个Proxy列表,用来指示后续请求消息的路由。
具体处理过程如下:
在一个请求消息的传输过程中,Proxy也可能会添加一个Record-Route头域,这样当消息到达被叫后里面就有会有0个或若干个Record-Route头域。被叫会将这些Record-Route头域并入自己的路由集,随后被叫在发送请求(和Record-Route属于同一个dialog里面)消息时就会使用该路由集构造一系列Route头域,以便对消息进行路由;
另外,被叫会像上面对待Via头域一样,将Record-Route头域全部原样copy到响应消息中返回给主叫。主叫收到响应消息后也会将这些Record-Route头域并入路由集,只是它会将其反序。该会话中的后续请求(和Record-Route属于同一个dialog里面)消息的Route头域就会通过路由集构造。
-
目的(Contact)
联系人或最终的目的地地址。不管是请求端还是目的端,当发起请求是目的地址就是Contact。请求消息或响应消息必须携带Contact头域,告诉对方自己的联系地址。
2.严格路由(Strict Routing)与松散路由(Louse Routing)
-
严格路由(Strict Routing):可以理解为比较“死板”的理由机制,这种路由机制在
SIP协议的前身RFC 2534中定义,其机制非常简单。
-
松散路由(Louse Routing):该路由机制较为灵活,也是SIP路由机制的灵魂所在,在RFC 3261中定义。
-
混合路由:如果在一条路由的路径中,既有严格路由又有松散路由,我们称之为混合路由。
如何判断一个Proxy是否为严格路由还是松散路由?
根据请求消息中的Route字段中设置的SIP URI一个lr的属性进行判断,当携带有“lr”属性就表明该Proxy为松散路由Proxy,否则就是严格路由。如:
Routesip:a.b.c.d;lr
这就是表示这个地址所在的Proxy是一个Loose Router。需要注意的是"lr"属性是在Proxy添加Record-Roud时设置的。
严格路由的处理步骤如下:
- 接收到的消息的request-URI必须是自己的URI;
- 把第一个Route头域“弹”出来,并把其中的URI作为新的request-URI;
- 然后把该消息路由到新的request-URI;
松散路由的处理步骤如下:
- Proxy首先会检查消息的request-URI是不是属于自己所负责的域。如果是,它就会通过定位服务将该地址“翻译”成具体的联系地址并以此替换掉原来的request-URI;否则,它不会动request-URI;
- Proxy检查路由表中的第一个地址是否为自己,如果是则从表中删除;
- Loose Router首先会检查Request URI是否为自己插入到路由表中的地址,
如果不是,则不作处理;如果是,则取出Route字段的最后一个地址作为Request URI地址,并从Route字段中删去最后一个地址; - Loose Router检查下一跳是否为Strict Router。如果不是,不处理;否则,将Request
URI插入到路由表表尾,并用下一跳地址Strict Router的地址)更新Request URI; - 如果路由表为空,则路由给Request URI;否则,路由到路由表表首地址。
3.总结
Via头域是为了给一个请求消息的响应消息留后路,而Record-Route就是为了给该请求消息之后的请求消息留后路。
Record-Route头域不用来路由,而只是起到传递信息的作用。
Route头域才是用来指示路由。
Contact头域是用来告诉对方我的联系地址。