XMPP 服务器 Openfire 的 Emoji 支持问题(进行部分修改)

当前最新版3.9.3已经可以支持Emoji 

----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

在为领航信息开发 eMessage 支持的时候,我们曾使用著名的开源 XMPP 服务器软件 Openfire。但在使用中遇到了几个问题,并通过修改源代码将这些问题解决掉了。接下来的几篇文章,我会介绍一下这些问题并讲述是如何解决掉的。

先介绍一下背景。XMPP 是一个开放的即时通讯协议,非常不错,有很多开源软件实现了 XMPP 协议,Openfire 算是实现得比较全的,而且安装配置比较容易。其他比较流行的开源 XMPP 服务器还有 Tigase 和 ejabberd。我们现在已经切换到了 ejabberd 上,毕竟 WhatsApp 最初也使用的是 ejabberd 嘛,呵呵。读者如果有兴趣,我也可以跟大家讲讲这几个 XMPP 服务器的区别和潜在问题。

问题描述

Emoji 现在基本已经成为一种工业事实标准,最早在日本流行,由日本的 DoKoMo 等运营商支持,后来苹果在 iOS 中支持了这一技术,最终变得全世界流行起来。Emoji 使用了一些 UNICODE 字符集中尚未定义的码位来表示一个个的表情图案。和一般的字体不同,应用程序在遇到这些字符的时候,需要使用位图来显示对应的表情图标,而不是直接使用字体中定义的字型来显示(当然,也可以用字体来显示 Emoji 字符,比如某些 Linux 控制台就可以显示部分 Emoji 字符)。前者可以是彩色的,而后者只能是某个特定的颜色。所以,要显示 Emoji 就要在应用程序中做扩展,在输出这些特定字符的时候做特殊处理,将其用对应的表情位图显示出来。让客户端应用支持 Emoji 并不是很难的工作,Android 的短信应用不支持 Emoji,但支持预先定义的特定字符序列来表示特定的表情,比如将“:)”显示成笑脸。处理这种字符序列的方法和处理 Emoji 表情的方法本质上一样的。

在使用 Openfire 作为 XMPP 服务器,将表示 Emoji 的 UNICODE 字符发送给其他用户的时候,就会出现问题。问题的原因在于 Emoji 使用的 UNICODE 字符集码位尚未被 UNICODE 标准化组织标准化,而 Openfire 会将 Emoji 字符看成是不符合标准的字符而直接忽略掉或者干脆断开客户端的连接。因此,要解决这个问题,其实相当容易,通过搜索引擎可以很快找到了解决方案。

解决方案

在 Openfire 3.8.2 版本源代码术中,修改 openfire_src/src/java/org/jivesoftware/openfire/net 目录下的 MXParser.java 文件的最后一个函数:

[java] view plaincopy

  1. /**
  2. * Makes sure that each individual character is a valid XML character.
  3. *
  4. * Note that when MXParser is being modified to handle multibyte chars correctly, this method needs to change (as
  5. * then, there are more codepoints to check).
  6. */
  7. @Override
  8. protected char more() throws IOException, XmlPullParserException {
  9. final char codePoint  = super.more(); // note - this does NOT return a codepoint now, but simply a (single byte) char
  10. er!
  11. if ((codePoint == 0x0) ||  // 0x0 is not allowed, but flash clients insist on sending this as the very first
  12. racter of a stream. We should stop allowing this codepoint after the first byte has been parsed.
  13. (codePoint == 0x9) ||
  14. (codePoint == 0xA) ||
  15. (codePoint == 0xD) ||
  16. ((codePoint >= 0x20) && (codePoint <= 0xD7FF)) ||
  17. ((codePoint >= 0xE000) && (codePoint <= 0xFFFD))) {
  18. return codePoint;
  19. throw new XmlPullParserException("Illegal XML character: " + Integer.parseInt(codePoint+"", 16));
  20. }

先看看这个函数的作用。如注释所言,这个函数用来判定特定字符是否是一个合法的 XML 字符。XML 一般要求按照 UTF8 编码的方式存储和传输字符,在程序处理时,会直接转换成 UNICODE 的 UCS 形式,这样便于程序做处理。

在 Linux 控制台上运行 $ man utf8 命令,你可以迅速知悉 UNICODE 码位范围以及和 UTF-8 编码之间的对应关系:

[plain] view plaincopy

  1. The following byte sequences are used to represent a character.  The sequence to be used depends on the  UCS  code
  2. number of the character:
  3. 0x00000000 - 0x0000007F:
  4. 0xxxxxxx
  5. 0x00000080 - 0x000007FF:
  6. 110xxxxx 10xxxxxx
  7. 0x00000800 - 0x0000FFFF:
  8. 1110xxxx 10xxxxxx 10xxxxxx
  9. 0x00010000 - 0x001FFFFF:
  10. 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx
  11. 0x00200000 - 0x03FFFFFF:
  12. 111110xx 10xxxxxx 10xxxxxx 10xxxxxx 10xxxxxx
  13. 0x04000000 - 0x7FFFFFFF:
  14. 1111110x 10xxxxxx 10xxxxxx 10xxxxxx 10xxxxxx 10xxxxxx

从上面的对应关系就可以知悉,UNICODE 可能的码位(code point 或者 code number)最大可以到 0x7FFFFFFF!而 Openfire 判断是否为合法 XML 字符的范围只到 0xFFFD!显然,不能显示正确处理 Emoji 字符的原因是 Openfire 将用来表示 Emoji 字符的那些码位范围给当成非法 XML 字符了。一种比较简单粗暴的修改办法是:

[java] view plaincopy

  1. /**
  2. * Makes sure that each individual character is a valid XML character.
  3. *
  4. * Note that when MXParser is being modified to handle multibyte chars correctly, this method needs to change (as
  5. * then, there are more codepoints to check).
  6. */
  7. @Override
  8. protected char more() throws IOException, XmlPullParserException {
  9. final char codePoint  = super.more(); // note - this does NOT return a codepoint now, but simply a (single byte) char
  10. r!
  11. if ((codePoint == 0x0) ||  // 0x0 is not allowed, but flash clients insist on sending this as the very first
  12. acter of a stream. We should stop allowing this codepoint after the first byte has been parsed.
  13. (codePoint == 0x9) ||
  14. (codePoint == 0xA) ||
  15. (codePoint == 0xD) ||
  16. ((codePoint >= 0x20) && (codePoint <= 0xFFFD)) ||
  17. ((codePoint >= 0x10000) && (codePoint <= 0x10FFFF))) {
  18. return codePoint;
  19. }
  20. throw new XmlPullParserException("Illegal XML character: " + Integer.parseInt(codePoint+"", 16));
  21. }

但马上有高手指出,这个方法太粗暴了,可能带来一些安全隐患(参见:http://community.igniterealtime.org/thread/48846),所以更加正确的方法是:

[java] view plaincopy

  1. @Override
  2. protected char more() throws IOException, XmlPullParserException {
  3. final char codePoint = super.more(); // note - this does NOT return a codepoint now, but simply a (double byte) character!
  4. boolean validCodepoint = false;
  5. boolean isLowSurrogate = Character.isLowSurrogate(codePoint);
  6. if ((codePoint == 0x0)|| // 0x0 is not allowed, but flash clients insist on sending this as the very first character of a stream. We should stop allowing this codepoint after the first byte has been parsed.
  7. (codePoint == 0x9) ||
  8. (codePoint == 0xA) ||
  9. (codePoint == 0xD)||
  10. ((codePoint >= 0x20) && (codePoint <= 0xD7FF)) ||
  11. ((codePoint >= 0xE000) && (codePoint <= 0xFFFD))) {
  12. validCodepoint = true;
  13. }
  14. else if (highSurrogateSeen) {
  15. if (isLowSurrogate) {
  16. validCodepoint = true;
  17. } else {
  18. throw new XmlPullParserException(
  19. "High surrogate followed by non low surrogate ‘0x"
  20. + String.format("%x", (int) codePoint) + "‘");
  21. }
  22. }
  23. else if (isLowSurrogate) {
  24. throw new XmlPullParserException("Low surrogate ‘0x "+ String.format("%x", (int) codePoint)+ " without preceeding high surrogate");
  25. }
  26. else if (Character.isHighSurrogate(codePoint)) {
  27. highSurrogateSeen = true;// Return here so that highSurrogateSeen is not reset
  28. return codePoint;
  29. }
  30. // Always reset high surrogate seen
  31. highSurrogateSeen = false;
  32. if (validCodepoint)
  33. return codePoint;
  34. throw new XmlPullParserException("Illegal XML character ‘0x"+ String.format("%x", (int) codePoint) + "‘");
  35. }

有了上述修改,你的 Openfire 服务器就可以正确处理 Emoji 了。Openfire 最新的版本是 3.9.1,估计已经修改掉这个问题了吧,但本人未确认。

在 MySQL 中存储 Emoji 字符

没想到,为了在 MySQL 数据库中保存 Emoji 字符,需要使用 MySQL 5.5 以上版本引入的 utf8mb4 的字符集。原来 MySQL 的 utf8 字符集只支持编码为一个、两个字节或者三个字节的情形,也就是 UNICODE  UCS 编码范围为 0 到 0xFFFD 这种情形。要支持超过这个范围的 UTF8 编码字符,就需要使用 MySQL 5.5 中引入的 utf8mb4 字符集。从名字中可以看出,这个字符集专门用来支持单个 UNICODE 的 UTF8 编码长度达到四个字节的情形。当然,超过也许也是可以的。至于为什么不能直接用 utf8 编码来兼容这些字符,我就不知道了,也许是历史原因吧。

大家可以用“mysql utf8mb4”为关键词搜索一下就知道如何设置/配置 mysql 来支持这个字符集了。但是,要让 openfire 能够和 mysql 正确打交道,还需要升级一下 openfire 使用的 JAVA mysql 数据库连接器(connnector),要升级到最新的版本。否则,如果使用老的 mysql 数据库连接器,会出现无法理解 utf8mb4 字符集的情形。

吐槽一下,我实在是不能理解为什么 MySQL 要引入 utf8mb4 这个字符集,难不成将来还需要引入 utf8mb6、utf8mb8 这样的字符集不成?哪位大侠可以帮我解答这个疑惑?

后记

俺是不太喜欢 JAVA 语言的,至今未能使用 JAVA 语言完整编写过一个程序,这实在是本人二十多年码农生涯的一大遗憾,但打打补丁这事儿还是可以做做的。下一篇文章给大家介绍一个针对 Openfire 服务器在集群环境下处理 SOCKS5 代理的问题,就修改了几行代码,但解决了一个大问题。

http://blog.csdn.net/ldwtill/article/details/23210835

时间: 2024-08-29 00:54:47

XMPP 服务器 Openfire 的 Emoji 支持问题(进行部分修改)的相关文章

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

XMPP(Extensible Messaging and Presence Protocol)是一种网络即时通讯协议,它基于XML,具有很强的扩展性,被广泛使用在即时通讯软件.网络游戏聊天.Web聊天及Web消息推送.移动设备的消息推送等场景,例如Google的GTalk.<英雄联盟LOL>游戏聊天模块. 由于在Web浏览器上的JavaScript不能直接处理TCP协议,所以XMPP服务器通常会提供BOSH(Bidirectional-streams Over Synchronous HTT

iOS 下配置XMPP 服务器openfire详解

一.下载并安装openfire 1.到http://www.igniterealtime.org/downloads/index.jsp下载最新openfire for mac版 比如:Openfire 3.8.1,下载后的文件:openfire_3_8_1.dmg 2.点击安装,并执行默认操作 3.启动openfire服务 在系统偏好设置的其他里,点击openfire偏好 启动后,点击Open Admin Console按钮,自动在浏览器中打开本地web配置页面http://localhost

转-XMPP(服务器Openfire)框架下,修改用户密码

1.先按照协议0077中修改密码的XML修改自己的 { /* <iq type='set' to='shakespeare.lit' id='change1'> <query xmlns='jabber:iq:register'> <username>bill</username> <password>newpass</password> </query> </iq> */ NSXMLElement *iq =

xmpp和OpenFire示例,即时聊天室,支持离线消息

让我说说为什么写这个博客,这是因为我在上周末的研究XMPP和OpenFire,从互联网上下载Demo,但跑不起来.它花了很长的时间.它被改造.抬高.篇博文也是希望后边学习XMPP和OpenFire的同学下载后直接执行.少走弯路了.时间就是金钱,不要花费不必要的时间,也希望大家都能有分享精神.,有问题能够发邮件给我([email protected]) 关于xmpp和openfire的资料请百度百科一下 下面展示一个聊天程序.所谓万事都要有Helloworld嘛,这个demo能够做为学习xmpp和

技术笔记:XMPP之openfire+spark+smack

在即时通信这个领域目前只找到一个XMPP协议,在其协议基础上还是有许多成熟的产品,而且是开源的.所以还是想在这个领域多多了解一下. XMPP协议:具体的概念我就不写了,毕竟这东西网上到处是.简单的说就是基于XML的一种协议.其解决了什么问题呢?就是给即时通讯制定了标准,大家只要遵守标准就可以完成即时通信的功能.有了标准的好处就是可以有各种不同的实现,大家在这个标准上发展自己的特长.而且还给即时通信提供了互联互通的基础.XMPP协议据网上说还是比较优秀的,表现就是google等大公司都在自己的即时

XMPP(一)-openfire服务端的安装和搭建

XMPP全称:可扩展通讯和表示协议 简介:可扩展通讯和表示协议 (XMPP) 可用于服务类实时通讯.表示和需求响应服务中的XML数据元流式传输.XMPP以Jabber协议为基础,而Jabber是即时通讯中常用的开放式协议.XMPP is the IETF's formalization of the base XML streaming protocols for instant messaging and presence developed within the Jabber open-so

基于XMPP的IOS聊天客户端程序(XMPP服务器架构)

最近看了关于XMPP的框架,以文本聊天为例,需要发送的消息为: <message type="chat" from="[email protected]" to="[email protected]">          <body>helloWord</body>      </message> 基中from是从哪个用户发送的消息,to是发给谁的消息,XMPP的用户都是以邮箱形式.body就是我们

Linux搭建XMPP服务器Tigase(Spark客户端测试)

Tigase是一个基于Java开发的XMPP服务器,类似于Openfire,可用于搭建一个即时通讯(Instant Messaging,简称IM)的平台. 1.准备 在安装Tigase之前,首先需要准备Java环境以及数据库(本文使用MySQL). Tigase下载:https://projects.tigase.org/projects/tigase-server/files 下载:tigase-server-5.2.2-b3463-dist-max.tar.gz,并解压缩文件: wget h

Android 基于XMPP Smack openfire 开发的聊天室

Android基于XMPP Smack openfire 开发的聊天室