项目下载地址:https://github.com/liuwenhaoCN/openfire/tree/websocket
老版本是需要安装websocket插件的。之前版本就不说了。
<script src="lib/XML/ObjTree.js"></script> <script src="lib/XML/jkl-dumper.js"></script>
自行百度下载新版本。我使用的版本上传至百度网盘:
链接: https://pan.baidu.com/s/1ALWJPAbb14yKtsY_CTj1Gg 提取码: vfc3
使用方法:
//json转xml function json2xml(jsonstring) { var xotree = new XML.ObjTree(); var xml = xotree.writeXML(jsonstring); //使用jkl-dumper.js中的formatXml方法将xml字符串格式化 //var xmlText = formatXml(xml); return xml; } //xml转json function xml2json(xmlstring) { //将xml字符串转为json var xotree = new XML.ObjTree(); var json = xotree.parseXML(xmlstring); //将json对象转为格式化的字符串 var dumper = new JKL.Dumper(); var jsonText = dumper.dump(json); return JSON.parse(jsonText); }
在http://192.168.2.11:9090/index.jsp 首页面可以看到以下信息:
服务器属性 服务器启动时间: 20 天, 23 小时, 22 分钟 -- started 2018-9-5 15:44:47 版本: Openfire 4.2.2 Openfire 主目录: /var/openfire 服务器名称: lvs-backup-1 环境 JVM 版本和供应商: 1.8.0_65 Oracle Corporation -- Java HotSpot(TM) 64-Bit Server VM 应用服务器: jetty/9.2.z-SNAPSHOT 主机名: lvs-backup-1 OS/硬件: Linux / amd64 语言环境/时区: zh_CN / 中国标准时间 (8 GMT) 服务器端口 所有的地址 7070 HTTP绑定 端口用于客户端使用不安全的HTTP方式访问 所有的地址 7443 端口用于网络上以SSL加密的通信,HTTP绑定端口用于客户端使用安全的HTTP方式访问
//地址 var host = "ws://192.168.2.11:7070/ws/"; //创建websocket, OpenFire是实现了WebSocket的子协议 websocket = new WebSocket(host, 'xmpp'); //连接成功建立的回调方法 websocket.onopen = wsopen; //连接发生错误的回调方法 websocket.onerror = wserror; //连接关闭的回调方法 websocket.onclose = wsclose; //收到消息时执行的方法 websocket.onmessage = wsmessage; 当websocket连接成功,就需要发起建立流的请求
发送xml
<?xml version="1.0" encoding="UTF-8" ?> <open to="lvs-backup-1" from="[email protected]" xmlns="urn:ietf:params:xml:ns:xmpp-framing" xml:lang="zh" version="1.0" />
方法
//域名 var to = "lvs-backup-1"; //JID var from = [email protected]; var xmlns = "urn:ietf:params:xml:ns:xmpp-framing"; var version = "1.0"; var xmllang = "zh"; var resource = "appClient"; //发起建立流的请求 function connwsopen() { var temp = { "open": { "-to": to, "-from": from, "-xmlns": xmlns, "-xml:lang": xmllang, "-version": version } }; //转化为xml var steam = json2xml(temp); websocket.send(steam); }
接收XML
服务器返回同意建立流
<open to='[email protected]' from='lvs-backup-1' id='1g0xa0mr98' xmlns='urn:ietf:params:xml:ns:xmpp-framing' xml:lang='zh' version='1.0'/>
安全验证的方式,使用PLAIN方式,这种方式是把用户名和密码通过BASE64加密后传输。
<stream:features xmlns:stream='http://etherx.jabber.org/streams'><mechanisms xmlns="urn:ietf:params:xml:ns:xmpp-sasl"><mechanism>PLAIN</mechanism><mechanism>SCRAM-SHA-1</mechanism><mechanism>CRAM-MD5</mechanism><mechanism>DIGEST-MD5</mechanism></mechanisms></stream:features>
使用PLAIN方式,需要对用户名和密码通过BASE64加密
字符串格式是:jid+password,以\0作为分隔符 var temp = username + "@" + to + "\0" + password; //Base64编码 var token = window.btoa(temp);
发送XML
<?xml version="1.0" encoding="UTF-8" ?> <auth xmlns="urn:ietf:params:xml:ns:xmpp-sasl" mechanism="PLAIN">MTUyNzUwNzYzNTNAbHZzLWJhY2t1cC0xAGUxMGFkYzM5NDliYTU5YWJiZTU2ZTA1N2YyMGY4ODNl</auth>
方法
//登录验证 function validatelogin() { //字符串格式是:jid+password,以\0作为分隔符 var temp = username + "@" + to + "\0" + password; //Base64编码 var token = window.btoa(temp); var message = { "auth": { "-xmlns": "urn:ietf:params:xml:ns:xmpp-sasl", "-mechanism": "PLAIN", "#text": token } }; websocket.send(json2xml(message)); }
接收XML
登录成功:
<success xmlns="urn:ietf:params:xml:ns:xmpp-sasl"/>
登录失败:
<failure xmlns="urn:ietf:params:xml:ns:xmpp-sasl"><not-authorized/></failure>
登录验证成功后,需要再发起一个新的流,新的流服务端会给予一些新的XML节点权限(、),这样才能发送一些其他功能的信息,比如发送消息,获取联系人列表,刚开始建立的第一个流失不能发送这些节点的。
在发起新的流之前,需要获取到第一个流的id,这个id就是与websocket的会话差不多。
发送XML
<?xml version="1.0" encoding="UTF-8" ?> <open xmlns="jabber:client" to="lvs-backup-1" version="1.0" from="[email protected]" id="1g0xa0mr98" xml:lang="zh" />
方法
id = "1g0xa0mr98"; 第一个流的id //发起新的流 function newopen() { var temp = { "open": { "-xmlns": "jabber:client", "-to": to, "-version": version, "-from": from, "-id": id, "-xml:lang": xmllang } }; //转化为xml var steam = json2xml(temp); websocket.send(steam); }
接收XML
同意打开新流
<open to='[email protected]' from='lvs-backup-1' id='1g0xa0mr98' xmlns='urn:ietf:params:xml:ns:xmpp-framing' xml:lang='zh' version='1.0'/>
接下来需要做bind操作
<stream:features xmlns:stream='http://etherx.jabber.org/streams'><bind xmlns='urn:ietf:params:xml:ns:xmpp-bind'/><session xmlns='urn:ietf:params:xml:ns:xmpp-session'><optional/></session><sm xmlns='urn:xmpp:sm:3'/></stream:features>
绑定操作,也需要id的,在绑定的时候,可以增加客户端的描述。
appClient:是app的客户端
发送XML
<?xml version="1.0" encoding="UTF-8" ?> <iq id="1g0xa0mr98" type="set"> <bind xmlns="urn:ietf:params:xml:ns:xmpp-bind"> <resource>appClient</resource> </bind> </iq>
方法
//bind操作 function bind() { var temp = { "iq": { "-id": id, "-type": "set", "bind": { "-xmlns": "urn:ietf:params:xml:ns:xmpp-bind", "resource":resource } } }; //转化为xml var steam = json2xml(temp); websocket.send(steam); }
接收
<iq xmlns="jabber:client" type="result" id="1g0xa0mr98" to="lvs-backup-1/8itpfrge4k"><bind xmlns="urn:ietf:params:xml:ns:xmpp-bind"><jid>[email protected]/appClient</jid></bind></iq>
发送XML
<?xml version="1.0" encoding="UTF-8" ?> <iq xmlns="jabber:client" id="1g0xa0mr98" type="set"> <session xmlns="urn:ietf:params:xml:ns:xmpp-session" /> </iq>
方法
//获取session function getsession() { var temp = { "iq": { "-xmlns": "jabber:client", "-id": id, "-type": "set", "session": {"-xmlns": "urn:ietf:params:xml:ns:xmpp-session"} } }; //转化为xml var steam = json2xml(temp); websocket.send(steam); }
接收
<iq xmlns="jabber:client" type="result" id="1g0xa0mr98" to="[email protected]/appClient"/>
这个时候,在web页面会话中看到的,账号还是离线状态。对应的消息也收不到。下一步就需要做上线的操作了。
发送XML
<?xml version="1.0" encoding="UTF-8" ?> <presence id="1g0xa0mr98"> <status>online</status> <priority>1</priority> </presence>
方法
priority:设置优先级的 //上线 function presence() { var temp = { "presence": { "-id": id, "status": "online", "priority": "1" } }; //转化为xml var steam = json2xml(temp); websocket.send(steam); }
接收XML
会把对应的好友发送过来
<presence xmlns="jabber:client" from="[email protected]" to="[email protected]" type="subscribe"/> <presence xmlns="jabber:client" from="[email protected]" to="[email protected]" type="subscribe"/>
这时候再看web中的会话,该账号就在线了。
发送XML
body:对应的消息内容 <?xml version="1.0" encoding="UTF-8" ?> <message type="chat" from="[email protected]" to="[email protected]"> <subject>标题</subject> <body>你好</body> </message>
方法
number:给谁发 message:消息内容 //发消息 function send(number,message) { var temp = { "message": { "-type":"chat", "-from": username+"@"+to, "-to": number+"@"+to, "subject":"标题", "body": message } }; //转化为xml var steam = json2xml(temp); websocket.send(steam); }
用户名: 15275076353 密码: e10adc3949ba59abbe56e057f20f883e 正在连接.. 连接已建立 发起建立流的请求: <?xml version="1.0" encoding="UTF-8" ?> <open to="lvs-backup-1" from="[email protected]" xmlns="urn:ietf:params:xml:ns:xmpp-framing" xml:lang="zh" version="1.0" /> 接收到的 <open to='[email protected]' from='lvs-backup-1' id='3r01w7mpew' xmlns='urn:ietf:params:xml:ns:xmpp-framing' xml:lang='zh' version='1.0'/> 接收到的 <stream:features xmlns:stream='http://etherx.jabber.org/streams'><mechanisms xmlns="urn:ietf:params:xml:ns:xmpp-sasl"> <mechanism>PLAIN</mechanism><mechanism>SCRAM-SHA-1</mechanism><mechanism>CRAM-MD5</mechanism><mechanism>DIGEST-MD5</mechanism></mechanisms> </stream:features> 登录验证: <?xml version="1.0" encoding="UTF-8" ?> <auth xmlns="urn:ietf:params:xml:ns:xmpp-sasl" mechanism="PLAIN">MTUyNzUwNzYzNTNAbHZzLWJhY2t1cC0xAGUxMGFkYzM5NDliYTU5YWJiZTU2ZTA1N2YyMGY4ODNl</auth> 接收到的 <success xmlns="urn:ietf:params:xml:ns:xmpp-sasl"/> 登录成功! 发起新的流: <?xml version="1.0" encoding="UTF-8" ?> <open xmlns="jabber:client" to="lvs-backup-1" version="1.0" from="[email protected]" id="3r01w7mpew" xml:lang="zh" /> 接收到的 <open to='[email protected]' from='lvs-backup-1' id='3r01w7mpew' xmlns='urn:ietf:params:xml:ns:xmpp-framing' xml:lang='zh' version='1.0'/> 接收到的 <stream:features xmlns:stream='http://etherx.jabber.org/streams'><bind xmlns='urn:ietf:params:xml:ns:xmpp-bind'/> <session xmlns='urn:ietf:params:xml:ns:xmpp-session'><optional/></session><sm xmlns='urn:xmpp:sm:3'/></stream:features> bind操作: <?xml version="1.0" encoding="UTF-8" ?> <iq id="3r01w7mpew" type="set"> <bind xmlns="urn:ietf:params:xml:ns:xmpp-bind"> <resource>appClient</resource> </bind> </iq> 接收到的 <iq xmlns="jabber:client" type="result" id="3r01w7mpew" to="lvs-backup-1/3r01w7mpew"><bind xmlns="urn:ietf:params:xml:ns:xmpp-bind"><jid>[email protected]/appClient</jid></bind></iq> 获取session: <?xml version="1.0" encoding="UTF-8" ?> <iq xmlns="jabber:client" id="3r01w7mpew" type="set"> <session xmlns="urn:ietf:params:xml:ns:xmpp-session" /> </iq> 接收到的 <iq xmlns="jabber:client" type="result" id="3r01w7mpew" to="[email protected]/appClient"/> 上线: <?xml version="1.0" encoding="UTF-8" ?> <presence id="3r01w7mpew"> <status>online</status> <priority>1</priority> </presence> 接收到的 <presence xmlns="jabber:client" from="[email protected]" to="[email protected]" type="subscribe"/> 接收到的 <presence xmlns="jabber:client" from="[email protected]" to="[email protected]" type="subscribe"/>