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

http://www.blogjava.net/wkkyo/archive/2012/02/13/369827.html


项目中需要用到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

 
 
< iq  to ="android@192.168.1.113/Spark 2.6.3"  type ="set"  id ="iq_13"  from ="iphone@192.168.1.113/xiff" >      < 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" >          < feature  xmlns ="http://jabber.org/protocol/feature-neg" >              < xmlns ="jabber:x:data"  type ="form" >                  < field  var ="stream-method"  type ="list-single" >                      < option >< value > http://jabber.org/protocol/bytestreams </ value ></ option >                      < option >< value > http://jabber.org/protocol/ibb </ value ></ option >                  </ field >              </ x >          </ feature >          < file  xmlns ="http://jabber.org/protocol/si/profile/file-transfer"  name ="img0545.png"  size ="152443" >< desc > send </ desc ></ file >      </ si > </ iq >

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

 

< iq  id ="iq_13"  to ="iphone@192.168.1.113/xiff"  from ="android@192.168.1.113/Spark 2.6.3"  type ="result" >
    
< si  xmlns ="http://jabber.org/protocol/si" >
        
< feature  xmlns ="http://jabber.org/protocol/feature-neg" >
            
< xmlns ="jabber:x:data"  type ="submit" >
                
< field  var ="stream-method" >
                     
< value > http://jabber.org/protocol/bytestreams </ value >
                     
< value > http://jabber.org/protocol/ibb </ value >
                
</ field >
             
</ x >
        
</ feature >
     
</ si >
</ iq >

 

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

 

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


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

 

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

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

 

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

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

 

< iq  type ="result"  id ="iq_17"  from ="proxy.192.168.1.113"  to ="iphone@192.168.1.113/xiff" >
    
< query  xmlns ="http://jabber.org/protocol/bytestreams" >
        
< streamhost  jid ="proxy.192.168.1.113"  host ="192.168.1.113"  port ="7777" />
    
</ query >
</ iq >

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

 

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

 

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

 

< iq  id ="iq_19"  to ="iphone@192.168.1.113/xiff"  type ="result"  from ="android@192.168.1.113/Spark 2.6.3" >
    
< query  xmlns ="http://jabber.org/protocol/bytestreams" >
        
< streamhost-used  jid ="proxy.192.168.1.113" />
    
</ query >
</ iq >

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

 

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

 
代理回复激活成功信息

 

< iq  type ="result"  id ="iq_21"  from ="proxy.192.168.1.113"  to ="iphone@192.168.1.113/xiff" />

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

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

 

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


### Openfire 客户端应用程序下载与使用教程 对于希望部署即时通讯解决方案的企业和个人而言,了解如何获取并配置Openfire客户端应用程序至关重要。通常情况下,当提及Openfire时,更多是指作为XMPP协议服务器实现的软件本身;然而,为了使用户能够连接到此服务器,还需要相应的客户端程序。 #### 获取适合的客户端应用 虽然官方并未提供特定于Openfire品牌的桌面或移动客户端,但任何兼容XMPP/Jabber标准的聊天工具都可以充当Openfire的有效前端界面[^1]。些流行的选项包括: - **Pidgin**: 支持多平台(Windows, Linux),具备广泛的插件支持。 - **Spark**: 由同家公司开发的专业级IM客户端,特别优化用于配合Openfire工作。 - **Dino IM Client**: 开源项目,注重隐私保护特性,在Linux环境中表现良好。 - 移动设备上可以考虑安装像Conversations这样的Android XMPP客户端。 #### 配置过程概述 旦选择了合适的客户端之后,则需按照如下指南完成基本设置: 1. 启动所选的应用程序,并创建新账户或者导入现有联系人列表; 2. 输入目标Openfire实例的域名/IP地址以及指定端口号,默认为5222 (TCP) 或者 5223 (SSL/TLS)[^2]; 3. 如果启用了加密认证方式(EXTERNAL),则可能还需上个人数字证书文件来辅助身份验证流程[^3]; 4. 提交表单后尝试登录测试连通性状况直至成功建立会话为止。 ```bash # 示例命令行参数递给 Spark 进行自动注册/登陆操作 spark --username=johndoe@example.com \ --password=your_password_here \ --server=localhost \ --port=5222 ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值