一、mosquitto broker篇
1.依赖环境安装:
yum install gcc-c++
yum install openssl-develyum install c-ares-develyum install libuuid-devle
注:某些系统可能自带这些依赖环境,直接过滤
2.websocket支持
git clone https://git.oschina.net/woniu201/libwebsockets.git
安装libwebsockets需要cmake命令,需要2.8以上版本,yum安装的可能是2.6的版本,需要手动下载2.8以上的版本进行安装,请自行百度或谷歌下载,本文档使用cmake-3.2.3-Linux-x86_64 ,安装后直接设置环境变量即可。
1)tar -zxvf cmake-3.2.3-Linux-x86_64.tar.gz
2)解压后添加环境变量,刷新环境变量
vi /etc/profile
注:需要卸载老版本的cmake ,yum remove cmake
3)查看版本
cmake --version
4)安装websocket
tar -zxvf libwebsockets-v1.5-stable.tar.gz
cd libwebsockets/
mkdir build
cd build/
cmake ..
make install
3.mosquitto安装
wget wget http://mosquitto.org/files/source/mosquitto-1.4.11.tar.gz
tar zxvf mosquitto-1.4.11.tar.gz
cd mosquitto-1.4.11
vi config.mk 修改如下三项值
#关闭tls验证
WITH_TLS:=no
WITH_TLS_PSK:=no
##开启websocket支持
WITH_WEBSOCKETS:=yes
make
make install

修改mosquitto.conf配置文件,添加websocket支持
cd /etc/mosquitto/
cp mosquitto.conf.example mosquitto.conf
vi mosquitto.conf
修改如下参数:
listener 81
protocol websockets

5.启动mosquitto
mosquitto -c /etc/mosquitto/mosquitto.conf

如出现如上错误
ln -s /usr/local/lib/libwebsockets.so.5 /usr/lib64/libwebsockets.so.5
二、java篇
javaclient端使用的 eclipse.paho 包,Pom配置如下:
<dependency>
<groupId>org.eclipse.paho</groupId>
<artifactId>org.eclipse.paho.client.mqttv3</artifactId>
<version>1.1.0</version>
</dependency>
<!-- 另外大牛写的mqtt客户端 -->
<dependency>
<groupId>org.fusesource.mqtt-client</groupId>
<artifactId>mqtt-client</artifactId>
<version>1.14</version>
</dependency>
1.client 代码:
import java.util.Scanner;

import org.eclipse.paho.client.mqttv3.MqttClient;
import org.eclipse.paho.client.mqttv3.MqttConnectOptions;
import org.eclipse.paho.client.mqttv3.MqttException;
import org.eclipse.paho.client.mqttv3.MqttMessage;
import org.eclipse.paho.client.mqttv3.MqttTopic;
import org.eclipse.paho.client.mqttv3.persist.MemoryPersistence;

/**
* [简要描述]:<br/>
* [详细描述]:<br/>
*
* @author xiaolinlin
* @version 1.0,2017年6月8日
* @since smile V100R001C00
*/
public class MqttTestClient
{
public static final String HOST = "tcp://ip:1881";

public static final String TOPIC = "mqtt/topic";

private static final String clientid = "client124";

private MqttClient client;

private MqttConnectOptions options;

private String userName = "admin";

private String passWord = "admin123";

private void start() throws MqttException
{
try
{
// host为主机名,clientid即连接MQTT的客户端ID,一般以唯一标识符表示,MemoryPersistence设置clientid的保存形式,默认为以内存保存
client = new MqttClient(HOST,clientid,new MemoryPersistence());
// MQTT的连接设置
options = new MqttConnectOptions();
// 设置是否清空session,这里如果设置为false表示服务器会保留客户端的连接记录,这里设置为true表示每次连接到服务器都以新的身份连接
options.setCleanSession(true);
// 设置连接的用户名
// options.setUserName(userName);
// 设置连接的密码
// options.setPassword(passWord.tochararray());
// 设置超时时间 单位为秒
options.setConnectionTimeout(10);
// 设置会话心跳时间 单位为秒 服务器会每隔1.5*20秒的时间向客户端发送个消息判断客户端是否在线,但这个方法并没有重连的机制
options.setKeepAliveInterval(20);
// 设置回调
MqttTopic topic = client.getTopic(TOPIC);
// setwill方法,如果项目中需要知道客户端是否掉线可以调用该方法。设置最终端口的通知消息
options.setwill(topic,"close".getBytes(),2,true);
client.connect(options);
client.setCallback(new ClientCallback(client,options));
// 订阅消息
int[] Qos = {1};
String[] topic1 = {TOPIC};
client.subscribe(topic1,Qos);
Scanner in = new Scanner(system.in);
String msg = "";
MqttMessage message = null;
while (true)
{
msg = in.nextLine();
if (msg.contains("colse"))
{
message = new MqttMessage(msg.getBytes());
break;
}
message = new MqttMessage(msg.getBytes());
topic.publish(message);
}
in.close();

}
catch (Exception e)
{
e.printstacktrace();
}
finally {
client.disconnect();
}
}

public static void main(String[] args) throws MqttException
{
MqttTestClient client = new MqttTestClient();
client.start();
}
}
// callback 类
import org.eclipse.paho.client.mqttv3.IMqttDeliveryToken;
import org.eclipse.paho.client.mqttv3.MqttCallback;
import org.eclipse.paho.client.mqttv3.MqttClient;
import org.eclipse.paho.client.mqttv3.MqttConnectOptions;
import org.eclipse.paho.client.mqttv3.MqttException;
import org.eclipse.paho.client.mqttv3.MqttMessage;
import org.eclipse.paho.client.mqttv3.MqttSecurityException;

/**
* [简要描述]:<br/>
* [详细描述]:<br/>
*
* @author xiaolinlin
* @version 1.0,2017年6月8日
* @since smile V100R001C00
*/
public class ClientCallback implements MqttCallback
{
private MqttClient client;
private MqttConnectOptions options;

public ClientCallback(MqttClient client,MqttConnectOptions options) {
this.client = client;
this.options = options;
}

@Override
public void connectionLost(Throwable cause)
{
// 连接丢失后,一般在这里面进行重连
System.out.println("连接断开,可以做重连");
try
{
client.connect(options);
}
catch (MqttSecurityException e)
{
// Todo Auto-generated catch block
e.printstacktrace();
}
catch (MqttException e)
{
// Todo Auto-generated catch block
e.printstacktrace();
}
// while(!sampleClient.isConnected()){
// try {
// Thread.sleep(1000);
// sampleClient.connect(connopts);
// //客户端每次上线都必须上传自己所有涉及的订阅关系,否则可能会导致消息接收延迟
// sampleClient.subscribe(topicFilters,qos);
// } catch (Exception e) {
// e.printstacktrace();
// }
// }
}

@Override
public void messageArrived(String topic,MqttMessage message) throws Exception
{
// subscribe后得到的消息会执行到这里面
System.out.println("接收消息主题 : " + topic);
System.out.println("接收消息Qos : " + message.getQos());
String msg = new String(message.getPayload());
System.out.println("接收消息到服务端内容 : " + msg);
if(msg.contains("close")) {
client.close();
}
}

@Override
public void deliveryComplete(IMqttDeliveryToken token)
{
System.out.println("deliveryComplete---------" + token.isComplete());
}

}
2:server端
import java.io.UnsupportedEncodingException;
import java.util.Scanner;

import org.eclipse.paho.client.mqttv3.MqttCallback;
import org.eclipse.paho.client.mqttv3.MqttClient;
import org.eclipse.paho.client.mqttv3.MqttClientPersistence;
import org.eclipse.paho.client.mqttv3.MqttConnectOptions;
import org.eclipse.paho.client.mqttv3.MqttDeliveryToken;
import org.eclipse.paho.client.mqttv3.MqttException;
import org.eclipse.paho.client.mqttv3.MqttMessage;
import org.eclipse.paho.client.mqttv3.MqttPersistenceException;
import org.eclipse.paho.client.mqttv3.MqttTopic;
import org.eclipse.paho.client.mqttv3.persist.MemoryPersistence;

/**
* [简要描述]:<br/>
* [详细描述]:<br/>
*
* @author xiaolinlin
* @version 1.0,2017年6月8日
* @since smile V100R001C00
*/
public class MqttServer
{
public static void main(String[] args) throws InterruptedException,MqttException
{
String serverURI = "tcp://ip:1881";
String clientId = "server001";
MqttClient client = null;
try
{
// MemoryPersistence设置clientid的保存形式,默认为以内存保存
MqttClientPersistence persistence = new MemoryPersistence();
// 异步 待研究
// MqttAsyncclient client = new MqttAsyncclient(serverURI,
// clientId,persistence );
client = new MqttClient(serverURI,clientId,persistence);
MqttConnectOptions options = new MqttConnectOptions();
// 设置是否清空session,这里如果设置为false表示服务器会保留客户端的连接记录,设置为true表示每次连接到服务器都以新的身份连接
options.setCleanSession(false);
// options.setUserName("admin");
// options.setPassword("admin123".tochararray());
// 超时时间 单位S
options.setConnectionTimeout(5);
// 设置会话心跳时间 单位S 服务器会每隔1.5*20秒的时间向客户端发送个消息判断客户端是否在线,但这个方法并没有重连的机制
options.setKeepAliveInterval(10);
// 自动重连
options.setAutomaticReconnect(true);

MqttCallback callback = new ReciveCallback(client,options);

// 建立链接
client.connect(options);
client.setCallback(callback);

String topicStr = "mqtt/topic";
MqttTopic topic = client.getTopic(topicStr);

String msg = "";
Scanner in = new Scanner(system.in);
while(true) {
msg = in.nextLine();
if(msg.contains("close")) {
pushMsg(topic,msg);
break;
}
pushMsg(topic,msg);
}
in.close();

}
catch (MqttException e)
{
// Todo Auto-generated catch block
e.printstacktrace();
}
finally {
client.close();
}
}

public static void pushMsg(MqttTopic topic,String msg)
{

MqttDeliveryToken token;
try
{
MqttMessage message = new MqttMessage();
message.setQos(1);
// 是否保留
message.setRetained(true);
byte[] payload = msg.getBytes("utf-8");
// 实际消息内容
message.setPayload(payload);
token = topic.publish(message);
// 等待完成
token.waitForCompletion();
// 打印完成状态
System.out.println(token.isComplete() + "========");
}
catch (MqttPersistenceException e)
{
// Todo Auto-generated catch block
e.printstacktrace();
}
catch (MqttException e)
{
// Todo Auto-generated catch block
e.printstacktrace();
}
catch (UnsupportedEncodingException e)
{
// Todo Auto-generated catch block
e.printstacktrace();
}

}

}
call back类
import org.eclipse.paho.client.mqttv3.IMqttDeliveryToken;
import org.eclipse.paho.client.mqttv3.MqttCallback;
import org.eclipse.paho.client.mqttv3.MqttClient;
import org.eclipse.paho.client.mqttv3.MqttConnectOptions;
import org.eclipse.paho.client.mqttv3.MqttException;
import org.eclipse.paho.client.mqttv3.MqttMessage;
import org.eclipse.paho.client.mqttv3.MqttSecurityException;

/**
* [简要描述]:<br/>
* [详细描述]:<br/>
*
* @author xiaolinlin
* @version 1.0,2017年6月8日
* @since smile V100R001C00
*/
public class ReciveCallback implements MqttCallback
{
public ReciveCallback(MqttClient client,MqttConnectOptions options) {
this.client = client;
this.options = options;
}
private MqttClient client;
private MqttConnectOptions options;

@Override
public void connectionLost(Throwable cause)
{
// 连接丢失后,一般在这里面进行重连
System.out.println("连接断开,可以做重连");
try
{
client.connect(options);
}
catch (MqttSecurityException e)
{
// Todo Auto-generated catch block
e.printstacktrace();
}
catch (MqttException e)
{
// Todo Auto-generated catch block
e.printstacktrace();
}
}

@Override
public void messageArrived(String topic,MqttMessage message) throws Exception
{
// subscribe后得到的消息会执行到这里面
System.out.println("接收消息主题 : " + topic);
System.out.println("接收消息Qos : " + message.getQos());
String msg = new String(message.getPayload());
System.out.println("接收消息客户端内容 : " + new String(message.getPayload()));
if(msg.contains("close")) {
client.close();
}
}

@Override
public void deliveryComplete(IMqttDeliveryToken token)
{
System.out.println("deliveryComplete---------" + token.isComplete());
}

}
三、js篇
HTML代码:
<!DOCTYPE html>
<html>
<head>
<Meta charset="utf-8">
<Meta http-equiv="X-UA-Compatible" content="IE=edge">
<Meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>聊天</title>
<link rel="stylesheet" href="">
<script language="JavaScript" src="jquery.js"></script>
<script language="JavaScript" src="mqttws31-min.js"></script>
<script type="text/javascript">
var topic = "mqtt/topic";
var hostname = "120.76.216.235";
var port = 81;
var reconnectTimeout = 2000;
var mqtt;
var options;
var username,password;

function MQTTconnect() {
var clientId = "ClientID_0022";
//var username = 'LTAItKYvBCx06Dbz';
mqtt = new Paho.MQTT.Client(hostname,port,clientId);
options = {
timeout: 3,
// useSSL: useTLS,
cleanSession: true,
onSuccess: onConnect,//if true(default) the client and server persistent state is deleted on successful connect.
onFailure: function (message) {
$('#status').val("Connection Failed: " + message.errorMessage + "retrying");
setTimeout(MQTTconnect,reconnectTimeout);
}
};
mqtt.onConnectionLost = onConnectionLost;
mqtt.onMessageArrived = onMessageArrived;
if (username) {
options.userName = username;
options.password = password;
}
console.log("username=" + username + " password=" + password);
mqtt.connect(options);
}
function onConnect() {
$('#status').val('Connected to ' + hostname + ':' + port);
// Connection succeeded; subscribe to our topic
mqtt.subscribe(topic,{qos: 0});
$('#topic').val(topic);
}
function onConnectionLost(response) {
setTimeout(MQTTconnect,reconnectTimeout);
$('#status').val("connection lost: " + response.errorMessage + ". Reconnecting");
};
function onMessageArrived(message) {
var topic = message.destinationName;
var payload = message.payloadString;
$('#ws').prepend('<li>' + topic + ' = ' + payload + '</li>');
};
function button_onclick(){
//这里写你要执行的语句
var tp = $('#sendtopic').val();
var val = $('#textsend').val();
if(tp==''||val=='')
{
alert("no aaa");
return;
}
var message = new Paho.MQTT.Message(val);
message.destinationName = tp;
message.qos=0;
mqtt.send(message);
// mqtt.publish(tp,val);
$('#ws').prepend('<li>' + tp + ' = ' + val+ '</li>');
};
$(document).ready(function() {
MQTTconnect();
});
</script>
</head>
<body>
<h1>Mosquitto Websockets</h1>
<div>
<div>Subscribed to <input type='text' id='topic' disabled />
Status: <input type='text' id='status' size="80" disabled />
</br>
publish to <input type='text' id='sendtopic' /> text <input type='text' id='textsend' size="80"/>
Status: <input type='button' value="send" id='btn' onclick="javascript:button_onclick()"/></div>
<ul id='ws' style="font-family: 'Courier New',Courier,monospace;"></ul>
</div>
</body>
</html>

引用的js自行下载,其中mqttws31-min.js 也是属于eclipse.paho 中的。

四、效果篇



附一些需要讨论的问题:
1.鉴权的考虑
2.点对点消息的考虑
3.性能考虑

Centos mosquitto mqtt支持 websocket javaclient jsclient的更多相关文章

  1. 五分钟学会HTML5的WebSocket协议

    这篇文章主要介绍了五分钟学会HTML5的WebSocket协议,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  2. 前端监听websocket消息并实时弹出(实例代码)

    这篇文章主要介绍了前端监听websocket消息并实时弹出,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  3. HTML5 WebSocket实现点对点聊天的示例代码

    这篇文章主要介绍了HTML5 WebSocket实现点对点聊天的示例代码的相关资料,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  4. ios – Websockets可以在移动电话上工作吗?

    相关地,我怀疑长轮询客户端可能是实现类似功能的好方法,但我想知道我可能遇到的移动特定问题.到目前为止,我已经读过长时间的轮询请求可能会对电池寿命产生相当大的影响.我还听说iOS以某种方式限制了对单个服务器的连接数量,这可能是个问题.有没有人在使用实时组件的移动应用程序上工作?

  5. ios-swift,objective-c协议实现

    作为隐式解开的可选项.

  6. ios – OSX上的AWS MQTT

    解决方法由于SSL握手问题,它是失败的.它正在检测到一个无效的证书.报告和解决了类似的问题here,引用相同的错误代码.由于p12文件中有多个身份,该问题被追溯到身份不匹配.在这种情况下,p12文件中有两个证书,但代码只读取第一个.我建议倾销.p12文件的内容,并确认证书.发布在这里审查.

  7. ios – 红蜘蛛代表没有被召集

    变量不是nil,我有一个很好的连接,url是正确的,但没有调用委托方法.我也正在实现WebSocketDelegate解决方法套接字应该是您的类的属性或变量,以确保它附近.如果仅在函数堆栈上分配它,它将超出范围,并且永远不会调用委托以下是我在项目中使用的代码,以防万一这是link到故事板,以防万一你想要

  8. swift 实现websocket与后台通信(swift 如何构建简单的json字符串)

    一个应用不可避免要与服务器进行通信,主要有,http与socket。http暂时不论,我们先看看socket下面衍生的websocket,今天我就把自己怎么利用websocket与服务器进行交互记录下来:首先你需要集成websocket到自己的项目,如果不明白如何集成,请看上一篇《swift集成websocket库》集成websocket到自己项目后还需要添加SwiftyJSON到自己项目,具体步骤和集成websocket一样。首先打开你项目,记得通过cocoapods生成的.xcworkspace文件打

  9. swift中string操作

    在swift中得string操作和OC稍有不同。一些基本操作苹果文档已经有描述。但是关于index的操作则没有提到。如果想删除或者得到字串,首先需要得到String.Index这个类型和Int不同,不能直接转换,所以需要用到advance函数。比如上面的例子,就是获取最后一个字符,然后删除之。

  10. 如何在Android上托管REST webservices?

    有没有人知道一个用Java编写的能够在Android上托管REST服务的开源Web服务器?

随机推荐

  1. 在airgapped(离线)CentOS 6系统上安装yum软件包

    我有一个CentOS6系统,出于安全考虑,它已经被空气泄漏.它可能从未连接到互联网,如果有,它很长时间没有更新.我想将所有.rpm软件包放在一个驱动器上,这样它们就可以脱机安装而无需查询互联网.但是,我在测试VM上遇到的问题是,即使指定了本地路径,yum仍然会挂起并尝试从在线存储库进行更新.另外,有没有办法使用yum-utils/yumdownloader轻松获取该包的所有依赖项和所有依赖项?目前

  2. centos – 命名在日志旋转后停止记录到rsyslog

    CentOS6.2,绑定9.7.3,rsyslog4.6.2我最近设置了一个服务器,我注意到在日志轮换后,named已停止记录到/var/log/messages.我认为这很奇怪,因为所有日志记录都是通过rsyslog进行的,并且named不会直接写入日志文件.这更奇怪,因为我在更新区域文件后命名了HUPed,但它仍然没有记录.在我停止并重新启动命名后,记录恢复.这里发生了什么?

  3. centos – 显示错误的磁盘大小

    对于其中一个磁盘,Df-h在我的服务器上显示错误的空白区域:Cpanel表明它只有34GB免费,但还有更多.几分钟前,我删除了超过80GB的日志文件.所以,我确信它完全错了.fdisk-l/dev/sda2也显示错误:如果没有格式化,我该怎么做才能解决这个问题?并且打开文件描述符就是它需要使用才能做到这一点.所以…使用“lsof”并查找已删除的文件.重新启动写入日志文件的服务,你很可能会看到空间可用.

  4. 如何在centos 6.9上安装docker-ce 17?

    我目前正在尝试在centOS6.9服务器上安装docker-ce17,但是,当运行yuminstalldocker-ce时,我收到以下错误:如果我用跳过的标志运行它我仍然得到相同的消息,有没有人知道这方面的方法?

  5. centos – 闲置工作站的异常负载平均值

    我有一个新的工作站,具有不寻常的高负载平均值.机器规格是:>至强cpu>256GB的RAM>4x512GBSSD连接到LSI2108RAID控制器我从livecd安装了CentOS6.564位,配置了分区,网络,用户/组,并安装了一些软件,如开发工具和MATLAB.在启动几分钟后,工作站负载平均值的值介于0.5到0.9之间.但它没有做任何事情.因此我无法理解为什么负载平均值如此之高.你能帮我诊断一下这个问题吗?

  6. centos – Cryptsetup luks – 检查内核是否支持aes-xts-plain64密码

    我在CentOS5上使用cryptsetupluks加密加密了一堆硬盘.一切都很好,直到我将系统升级到CentOS6.现在我再也无法安装磁盘了.使用我的关键短语装载:我收到此错误:在/var/log/messages中:有关如何装载的任何想法?找到解决方案问题是驱动器使用大约512个字符长的交互式关键短语加密.出于某种原因,CentOS6中的新内核模块在由旧版本创建时无法正确读取512个字符的加密密钥.似乎只会影响内核或cryptsetup的不同版本,因为在同一系统上创建和打开时,512字符的密钥将起作用

  7. centos – 大量ssh登录尝试

    22个我今天登录CentOS盒找到以下内容这是过去3天内的11次登录尝试.WTF?请注意,这是我从我的提供商处获得的全新IP,该盒子是全新的.我还没有发布任何关于此框的内容.为什么我会进行如此大量的登录尝试?是某种IP/端口扫描?基本上有4名匪徒,其中2名来自中国,1名来自香港,1名来自Verizon.这只发生在SSH上.HTTP上没有问题.我应该将罪魁祸首子网路由吗?你们有什么建议?

  8. centos – kswap使用100%的CPU,即使有100GB的RAM也可用

    >Linux内核是否应该足够智能,只需从内存中清除旧缓存页而不是启动kswap?

  9. centos – Azure将VM从A2 / 3调整为DS2 v2

    我正在尝试调整前一段时间创建的几个AzureVM,从基本的A3和标准A3到标准的DS2v2.我似乎没有能力调整到这个大小的VM.必须从头开始重建服务器会有点痛苦.如果它有所不同我在VM中运行CentOS,每个都有一个带有应用程序和操作系统的磁盘.任何人都可以告诉我是否可以在不删除磁盘的情况下删除VM,创建新VM然后将磁盘附加到新VM?

  10. centos – 广泛使用RAM时服务器计算速度减慢

    我在非常具体的情况下遇到服务器速度下降的问题.事实是:>1)我使用计算应用WRF>2)我使用双XeonE5-2620v3和128GBRAM(NUMA架构–可能与问题有关!

返回
顶部