Strophe.js连接XMPP服务器Openfire、Tigase实现Web私聊、群聊(MUC)

本文主要是介绍Strophe.js连接XMPP服务器Openfire、Tigase实现Web私聊、群聊(MUC),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

http://ju.outofmemory.cn/entry/110340


β
登录 注册
阿里云服务器,值得信赖    京东招聘运维工程师,待遇丰厚    服务器只买阿里云140万企业与开发者的信赖

Strophe.js连接XMPP服务器Openfire、Tigase实现Web私聊、群聊(MUC)

姜糖水 2015-01-14 1695 阅读
web开发

XMPP(Extensible Messaging and Presence Protocol)是一种网络即时通讯协议,它基于XML,具有很强的扩展性,被广泛使用在即时通讯软件、网络游戏聊天、Web聊天及Web消息推送、移动设备的消息推送等场景,例如Google的GTalk、《英雄联盟LOL》游戏聊天模块。

由于在Web浏览器上的JavaScript不能直接处理TCP协议,所以XMPP服务器通常会提供BOSH(Bidirectional-streams Over Synchronous HTTP)接口,通过HTTP长轮训(long-polling)可以实现Web浏览器即时聊天。Strophe.js是一个通过BOSH连接Web浏览器和XMPP服务器的工具库。

XMPP协议简介:

XMPP服务器和客户端之间,是通过XML节(XML Stanza)来进行通讯。其中有三种非常重要的XML Stanza类型:<message>、<presence>、<iq>。

<message>:

聊天消息的发送和接收就是通过message节来实现。例如xxg1@host发送一条信息”你好”给xxg2@host,xxg1@host客户端会将下面的这段XML发送到XMPP服务器,服务器再推送给xxg2@host客户端。其中<message>的from属性是发送者,to属性是接收者,<body>子元素的内容就是聊天信息。

<message from="xxg1@host" to="xxg2@host" type="chat"><body>你好</body>
</message>

<presence>:

可用于表明用户的状态,例如用户状态改变成“Do not disturb”(“请勿打扰”),会向服务器发送:

<presence from="xxg@host"><status>Do not disturb</status><priority>0</priority><show>dnd</show>
</presence>

<iq>:

iq即Info/Query,采用“请求-响应”机制,类似于HTTP的机制。下面的例子是客户端通过<iq>请求获取联系人,XMPP服务器将结果返回:

客户端请求获取联系人:

<iq from='xxg@host' id='bv1bs71f' type='get'><query xmlns='jabber:iq:roster'/>
</iq>

服务器结果返回:

<iq to='xxg@host' id='bv1bs71f' type='result'><query xmlns='jabber:iq:roster'><item jid='xxg2@host'/><item jid='xxg3@host'/></query>
</iq>

搭建XMPP服务器:

在实现Web浏览器聊天之前,首先要搭建一个XMPP服务器。例如Openfire、Tigase、Ejabberd是常用的XMPP服务器。其中Openfire、Tigase是基于Java实现,Ejabberd是Erlang实现。虽然实现的语言不同,但是都遵循XMPP协议,所以使用其中任意一个XMPP服务器即可。

下面以Openfire和Tigase为例。

Openfire可以自动化搭建很方便,本文不再介绍。Tigase的搭建可以参考我的另一篇博文:Linux搭建XMPP服务器Tigase(Spark客户端测试)。

XMPP服务器通常会实现BOSH扩展,下面是Openfire和Tigase的BOSH默认URL:

Openfire:http://host:7070/http-bind
Tigase:http://host:5280

在使用Strophe.js的时候,需要使用对应的HTTP地址才能连接上XMPP服务器。

如果使用Opnefire,还需要在管理后台配置一下:

Strophe.js:

下载:http://strophe.im/strophejs/

实现Web私聊:

私聊比较简单,聊天信息是通过上面介绍的<message>来进行传递交换。例如接收到一条别人发来的聊天信息,即接收一个<message>元素,发送给别人一条聊天信息,即发送一个<message>元素。

HTML:

<!DOCTYPE html>
<html>
<head><script src="http://cdn.bootcss.com/jquery/1.9.1/jquery.min.js"></script><script src="http://cdn.bootcss.com/strophe.js/1.1.3/strophe.min.js"></script><script src="http://www.cnphp6.com/archives/test.js"></script>
</head>
<body>JID:<input type="text" id="input-jid"><br>密码:<input type="password" id="input-pwd"><br><button id="btn-login">登录</button><div id="msg" style="height: 400px; width: 400px; overflow: scroll;"></div>联系人JID:<input type="text" id="input-contacts"><br>消息:<br><textarea id="input-msg" cols="30" rows="4"></textarea><br><button id="btn-send">发送</button>
</body>
</html>

JavaScript(test.js):

// XMPP服务器BOSH地址
var BOSH_SERVICE = 'http://host:5280';// XMPP连接
var connection = null;// 当前状态是否连接
var connected = false;// 当前登录的JID
var jid = "";// 连接状态改变的事件
function onConnect(status) {console.log(status)if (status == Strophe.Status.CONNFAIL) {alert("连接失败!");} else if (status == Strophe.Status.AUTHFAIL) {alert("登录失败!");} else if (status == Strophe.Status.DISCONNECTED) {alert("连接断开!");connected = false;} else if (status == Strophe.Status.CONNECTED) {alert("连接成功,可以开始聊天了!");connected = true;// 当接收到<message>节,调用onMessage回调函数connection.addHandler(onMessage, null, 'message', null, null, null);// 首先要发送一个<presence>给服务器(initial presence)connection.send($pres().tree());}
}// 接收到<message>
function onMessage(msg) {// 解析出<message>的from、type属性,以及body子元素var from = msg.getAttribute('from');var type = msg.getAttribute('type');var elems = msg.getElementsByTagName('body');if (type == "chat" && elems.length > 0) {var body = elems[0];$("#msg").append(from + ":<br>" + Strophe.getText(body) + "<br>")}return true;
}$(document).ready(function() {// 通过BOSH连接XMPP服务器$('#btn-login').click(function() {if(!connected) {connection = new Strophe.Connection(BOSH_SERVICE);connection.connect($("#input-jid").val(), $("#input-pwd").val(), onConnect);jid = $("#input-jid").val();}});// 发送消息$("#btn-send").click(function() {if(connected) {if($("#input-contacts").val() == '') {alert("请输入联系人!");return;}// 创建一个<message>元素并发送var msg = $msg({to: $("#input-contacts").val(), from: jid, type: 'chat'}).c("body", null, $("#input-msg").val());connection.send(msg.tree());$("#msg").append(jid + ":<br>" + $("#input-msg").val() + "<br>");$("#input-msg").val('');} else {alert("请先登录!");}});
});

修改JavaScript代码中的BOSH_SERVICE,用浏览器打开HTML文件,登录后即可聊天:

实现Web群聊:

XMPP群聊通过XMPP协议的MUC(Multi-User Chat)扩展实现。

Openfire默认支持MUC,但是Tigase服务器默认不支持MUC,需要在init.properties文件中加入以下粗体部分的配置项:

config-type=–gen-config-def
–admins=admin@host
–virt-hosts = host
–debug=server
–user-db=mysql
–user-db-uri = jdbc:mysql://localhost:3306/tigasedb?user=root&password=xxx
–comp-name-1 = muc
–comp-class-1 = tigase.muc.MUCComponent
–external = muc.devel.tigase.org:muc-pass:connect:5270:devel.tigase.org:accept

创建房间:

创建房间实际上可以写在代码中,但是本文为了方便,就使用XMPP客户端Spark或者其他工具来创建。

首先使用Spark任意登录一个用户,下图是Spark创建房间的步骤:

如果使用的是Tigase,默认创建的房间是加锁的,别的用户无法进入,需要对房间解锁。Spark进入房间后,点击下面的设置按钮,然后不用更改任何设置,直接更新即可解锁房间(虽然没有修改任何配置,但是更新时会发送一个<iq>给服务器,这个<iq>解锁了房间,参考http://xmpp.org/extensions/xep-0045.html#createroom-instant):

另外,如果使用Openfire可以直接使用管理后台来创建:

加入房间:

房间创建好了之后,就有有对应的房间JID:

加入房间可以通过发送一个<presence>来实现(实际上如果房间不存在下面的这条<presence>也会创建房间,但是创建的房间默认加锁,还需要发送一条<iq>解锁,所以本文就直接用Spark创建房间):

<presence from='xxg@host' to='xxgroom@muc.host/xxg'><x xmlns='http://jabber.org/protocol/muc'/>
</presence>

属性to=’xxgroom@muc.host/xxg’中,xxgroom@muc.host表示房间JID,xxg表示在房间的昵称。

聊天:

和私聊一样,群聊也是通过<message>来实现,不同的是<message>的type属性,私聊是”chat”,而群聊是”groupchat”,另外,to属性即为房间JID,这样一条聊天消息就会发送给房间中的所有人。

<message from='xxg@host' to='myroom@muc.host' type='groupchat'><body>大家好!</body>
</message>

实现:

HTML:

<!DOCTYPE html>
<html>
<head><script src="http://cdn.bootcss.com/jquery/1.9.1/jquery.min.js"></script><script src="http://cdn.bootcss.com/strophe.js/1.1.3/strophe.min.js"></script><script src="http://www.cnphp6.com/archives/test2.js"></script>
</head>
<body>JID:<input type="text" id="input-jid"><br>密码:<input type="password" id="input-pwd"><br><button id="btn-login">登录</button><div id="msg" style="height: 400px; width: 400px; overflow: scroll;"></div><br>消息:<br><textarea id="input-msg" cols="30" rows="4"></textarea><br><button id="btn-send">发送</button>
</body>
</html>

JavaScript(test2.js):

// XMPP服务器BOSH地址
var BOSH_SERVICE = 'http://host:5280';// 房间JID
var ROOM_JID = 'xxgroom@muc.host';// XMPP连接
var connection = null;// 当前状态是否连接
var connected = false;// 当前登录的JID
var jid = "";// 连接状态改变的事件
function onConnect(status) {if (status == Strophe.Status.CONNFAIL) {alert("连接失败!");} else if (status == Strophe.Status.AUTHFAIL) {alert("登录失败!");} else if (status == Strophe.Status.DISCONNECTED) {alert("连接断开!");connected = false;} else if (status == Strophe.Status.CONNECTED) {alert("连接成功,可以开始聊天了!");connected = true;// 当接收到<message>节,调用onMessage回调函数connection.addHandler(onMessage, null, 'message', null, null, null);// 首先要发送一个<presence>给服务器(initial presence)connection.send($pres().tree());// 发送<presence>元素,加入房间connection.send($pres({from: jid,to: ROOM_JID + "/" + jid.substring(0,jid.indexOf("@"))}).c('x',{xmlns: 'http://jabber.org/protocol/muc'}).tree());}
}// 接收到<message>
function onMessage(msg) {console.log(msg);// 解析出<message>的from、type属性,以及body子元素var from = msg.getAttribute('from');var type = msg.getAttribute('type');var elems = msg.getElementsByTagName('body');if (type == "groupchat" && elems.length > 0) {var body = elems[0];$("#msg").append(from.substring(from.indexOf('/') + 1) + ":<br>" + Strophe.getText(body) + "<br>")}return true;
}$(document).ready(function() {// 通过BOSH连接XMPP服务器$('#btn-login').click(function() {if(!connected) {connection = new Strophe.Connection(BOSH_SERVICE);connection.connect($("#input-jid").val(), $("#input-pwd").val(), onConnect);jid = $("#input-jid").val();}});// 发送消息$("#btn-send").click(function() {if(connected) {// 创建一个<message>元素并发送var msg = $msg({to: ROOM_JID, from: jid, type: 'groupchat'}).c("body", null, $("#input-msg").val());connection.send(msg.tree());$("#input-msg").val('');} else {alert("请先登录!");}});
});

创建好房间,修改JavaScript代码中的BOSH_SERVICE和ROOM_JID,用浏览器打开HTML文件,登录后即可群聊:

另外,Strophe.js还有一个专门的MUC插件,有兴趣的同学可以自己研究下:https://github.com/strophe/strophejs-plugins/tree/master/muc。

web开发
作者:姜糖水
又一个码农站点
原文地址: Strophe.js连接XMPP服务器Openfire、Tigase实现Web私聊、群聊(MUC), 感谢原作者分享。
→二叉排序树的插入与删除 ←基于jQuery的美食时间轴焦点图插件

发表评论

© 2015 内存溢出
复制代码
保存代码

这篇关于Strophe.js连接XMPP服务器Openfire、Tigase实现Web私聊、群聊(MUC)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/402726

相关文章

JS常用组件收集

收集了一些平时遇到的前端比较优秀的组件,方便以后开发的时候查找!!! 函数工具: Lodash 页面固定: stickUp、jQuery.Pin 轮播: unslider、swiper 开关: switch 复选框: icheck 气泡: grumble 隐藏元素: Headroom

服务器集群同步时间手记

1.时间服务器配置(必须root用户) (1)检查ntp是否安装 [root@node1 桌面]# rpm -qa|grep ntpntp-4.2.6p5-10.el6.centos.x86_64fontpackages-filesystem-1.41-1.1.el6.noarchntpdate-4.2.6p5-10.el6.centos.x86_64 (2)修改ntp配置文件 [r

W外链微信推广短连接怎么做?

制作微信推广链接的难点分析 一、内容创作难度 制作微信推广链接时,首先需要创作有吸引力的内容。这不仅要求内容本身有趣、有价值,还要能够激起人们的分享欲望。对于许多企业和个人来说,尤其是那些缺乏创意和写作能力的人来说,这是制作微信推广链接的一大难点。 二、精准定位难度 微信用户群体庞大,不同用户的需求和兴趣各异。因此,制作推广链接时需要精准定位目标受众,以便更有效地吸引他们点击并分享链接

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

在JS中的设计模式的单例模式、策略模式、代理模式、原型模式浅讲

1. 单例模式(Singleton Pattern) 确保一个类只有一个实例,并提供一个全局访问点。 示例代码: class Singleton {constructor() {if (Singleton.instance) {return Singleton.instance;}Singleton.instance = this;this.data = [];}addData(value)

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount