现在的位置: 首页 > 综合 > 正文

openfire客户端文件传输学习笔记

2013年10月05日 ⁄ 综合 ⁄ 共 6786字 ⁄ 字号 评论关闭

http://wkkyo.iteye.com/blog/1403465

项目中需要用到openfire的文件传输,但是客户端使用flex,官方提供的xiff包中并没有封装文件传输的功能,没办法,研究了几天,在google和官方smock源码的帮助下终于实现了xiff下的文件传输,在这里做个总结。

openfire服务器是基于xmpp协议的,XMPP支持两种文件流传输协议,SOCKS5 Bytestreams和 In-Band Bytestreams,SOCKS5是直接发送二进制流,而IBB是将文件转成base64码进行然后用message的形式进行传输,我这里仅实现了SOCKS5的文件代理传输。
SOCKS5文件传输需要用到两个协议,XEP-0065和XEP-0096
XEP-0096定义文件传输协议,提供了一个模块化框架使能交换被传输文件的信息以及参数的协商,也就是在传输文件之前协商将要传输的文件信息。
XEP-0065定义SOCKS5流传输标准协议,提供用于在任意两个XMPP用户之间建立字节流并进行文件传输。
根据我的理解,文件传输的过程分为协商,建立socks5连接,二进制传输这三个阶段
协商的过程最复杂,然后是建立连接,传输就比较简单,下面一个一个来讲
协商包括初始方、目标方、代理方,初始方就是发送文件方,目标方即文件接收方,代理方是socks5代理服务器,

协商过程就是三方互相发送xml来交换信息的过程,通俗点就是三个人沟通一下传什么文件和怎么传文件。
首先遵循XMP-0096协议,初始方给目标方发送包含文件信息的xml

Xml代码
复制代码

收藏代码
  1. <iq
    to="android@192.168.1.113/Spark 2.6.3"
    type="set"
    id="iq_13"
    from="iphone@192.168.1.113/xiff"> 
  2.     <si
    profile="http://jabber.org/protocol/si/profile/file-transfer"
    mime-type="text/plain"
    id="82B0C697-C1DE-93F9-103E-481C8E7A3BD8"
    xmlns="http://jabber.org/protocol/si"> 
  3.         <feature
    xmlns="http://jabber.org/protocol/feature-neg"> 
  4.             <x
    xmlns="jabber:x:data"
    type="form"> 
  5.                 <field
    var="stream-method"
    type="list-single"> 
  6.                     <option><value>http://jabber.org/protocol/bytestreams</value></option> 
  7.                     <option><value>http://jabber.org/protocol/ibb</value></option> 
  8.                 </field> 
  9.             </x> 
  10.         </feature> 
  11.         <file
    xmlns="http://jabber.org/protocol/si/profile/file-transfer"
    name="img0545.png"
    size="152443"><desc>send</desc></file> 
  12.     </si> 
  13. </iq> 

目标方接收到信息后发送回执,表示同意接收文件

Xml代码
复制代码

收藏代码
  1. <iq
    id="iq_13"
    to="iphone@192.168.1.113/xiff"
    from="android@192.168.1.113/Spark 2.6.3"
    type="result"> 
  2.     <si
    xmlns="http://jabber.org/protocol/si"> 
  3.         <feature
    xmlns="http://jabber.org/protocol/feature-neg"> 
  4.             <x
    xmlns="jabber:x:data"
    type="submit"> 
  5.                 <field
    var="stream-method"> 
  6.                      <value>http://jabber.org/protocol/bytestreams</value> 
  7.                      <value>http://jabber.org/protocol/ibb</value> 
  8.                 </field> 
  9.              </x> 
  10.         </feature> 
  11.      </si> 
  12. </iq> 

这时进入XEP-0065协议阶段
初始方给服务器发送信息,请求提供代理服务器

Xml代码
复制代码

收藏代码
  1. <iq
    id="iq_15"
    type="get"><query
    xmlns="http://jabber.org/protocol/disco#items"
    /></iq> 

服务器回复信息,告知可用的代理

Xml代码
复制代码

收藏代码
  1. <iq
    type="result"
    id="iq_15"
    to="iphone@192.168.1.113/xiff"> 
  2.     <query
    xmlns="http://jabber.org/protocol/disco#items"> 
  3.         <item
    jid="proxy.192.168.1.113"
    name="Socks 5 Bytestreams Proxy"/> 
  4.         <item
    jid="pubsub.192.168.1.113"
    name="Publish-Subscribe service"/> 
  5.         <item
    jid="conference.192.168.1.113"
    name="公共房间"/> 
  6.         <item
    jid="search.192.168.1.113"
    name="User Search"/> 
  7.     </query> 
  8. </iq> 

这里选择name=“Socks 5 Bytestreams Proxy”的代理,初始方给这个代理发送信息获取代理连接信息

Xml代码
复制代码

收藏代码
  1. <iq
    id="iq_17"
    to="proxy.192.168.1.113"
    type="get"><query
    xmlns="http://jabber.org/protocol/bytestreams"
    /></iq> 

代理方回复信息,告知初始方代理的jid、IP、端口等信息

Xml代码
复制代码

收藏代码
  1. <iq
    type="result"
    id="iq_17"
    from="proxy.192.168.1.113"
    to="iphone@192.168.1.113/xiff"> 
  2.     <query
    xmlns="http://jabber.org/protocol/bytestreams"> 
  3.         <streamhost
    jid="proxy.192.168.1.113"
    host="192.168.1.113"
    port="7777"/> 
  4.     </query> 
  5. </iq> 

初始方收到代理信息后将代理的信息发送给目标方

Xml代码
复制代码

收藏代码
  1. <iq
    to="android@192.168.1.113/Spark 2.6.3"
    type="set"
    id="iq_19"
    from="iphone@192.168.1.113/xiff"> 
  2.     <query
    xmlns="http://jabber.org/protocol/bytestreams"
    mode="tcp"
    sid="82B0C697-C1DE-93F9-103E-481C8E7A3BD8"> 
  3.         <streamhost
    port="7777"
    host="192.168.1.113"
    jid="proxy.192.168.1.113"
    /> 
  4.     </query> 
  5. </iq> 

然后就进入连接阶段,也就是初始方和目标方分别和代理建立socks5连接的过程。(关于SOCKS5协议连接,我之后会补充)。
目标方收到代理信息后和代理建立socket连接(使用SOCKS5协议连接),连接成功后通知初始方使用的代理jid

Xml代码
复制代码

收藏代码
  1. <iq
    id="iq_19"
    to="iphone@192.168.1.113/xiff"
    type="result"
    from="android@192.168.1.113/Spark 2.6.3"> 
  2.     <query
    xmlns="http://jabber.org/protocol/bytestreams"> 
  3.         <streamhost-used
    jid="proxy.192.168.1.113"/> 
  4.     </query> 
  5. </iq> 

初始方开始与代理建立socket连接(也使用SOCKS5协议),连接成功后给代理发送请求,要求激活文件流

<iq
to="proxy.192.168.1.113"
type="set"
id="iq_21"
from="iphone@192.168.1.113/xiff"> 
  1. <query
    xmlns="http://jabber.org/protocol/bytestreams"
    sid="82B0C697-C1DE-93F9-103E-481C8E7A3BD8"> 
  2. <activate>android@192.168.1.113/Spark 2.6.3</activate> 
  3. </query> 
  4. </iq> 

代理回复激活成功信息

Xml代码
复制代码

收藏代码
  1. <iq
    type="result"
    id="iq_21"
    from="proxy.192.168.1.113"
    to="iphone@192.168.1.113/xiff"/> 

初始方收到回复信息后就进入二进制流传输阶段,这时就可以开始发送二进制流了

等初始方将流发送完毕后把socket流关闭传输就完成了文件的传输。

注意:type为result的回复信息使用的id一定要和请求的信息id一样。

 

【上篇】
【下篇】

抱歉!评论已关闭.