commit
38d32cf529
14
CHANGELOG.md
14
CHANGELOG.md
|
@ -2,6 +2,20 @@
|
||||||
emqttd ChangeLog
|
emqttd ChangeLog
|
||||||
==================
|
==================
|
||||||
|
|
||||||
|
0.9.3-alpha (2015-07-25)
|
||||||
|
-------------------------
|
||||||
|
|
||||||
|
Wiki: [Bridge](https://github.com/emqtt/emqttd/wiki/Bridge)
|
||||||
|
|
||||||
|
Improve: emqttd_protocol.hrl to define 'QOS_I'
|
||||||
|
|
||||||
|
Improve: emqttd_pubsub to add subscribe/2 API
|
||||||
|
|
||||||
|
Improve: ./bin/emqttd_ctl to support new bridges command
|
||||||
|
|
||||||
|
Bugfix: issue #206 - Cannot bridge two nodes
|
||||||
|
|
||||||
|
|
||||||
0.9.2-alpha (2015-07-18)
|
0.9.2-alpha (2015-07-18)
|
||||||
-------------------------
|
-------------------------
|
||||||
|
|
||||||
|
|
|
@ -49,6 +49,26 @@
|
||||||
|
|
||||||
-type mqtt_qos() :: ?QOS_0 | ?QOS_1 | ?QOS_2.
|
-type mqtt_qos() :: ?QOS_0 | ?QOS_1 | ?QOS_2.
|
||||||
|
|
||||||
|
-type mqtt_qos_name() :: qos0 | at_most_once |
|
||||||
|
qos1 | at_least_once |
|
||||||
|
qos2 | exactly_once.
|
||||||
|
|
||||||
|
-define(QOS_I(Name),
|
||||||
|
begin
|
||||||
|
(case Name of
|
||||||
|
?QOS_0 -> ?QOS_0;
|
||||||
|
qos0 -> ?QOS_0;
|
||||||
|
at_most_once -> ?QOS_0;
|
||||||
|
?QOS_1 -> ?QOS_1;
|
||||||
|
qos1 -> ?QOS_1;
|
||||||
|
at_least_once -> ?QOS_1;
|
||||||
|
?QOS_2 -> ?QOS_2;
|
||||||
|
qos2 -> ?QOS_2;
|
||||||
|
exactly_once -> ?QOS_2
|
||||||
|
end)
|
||||||
|
end).
|
||||||
|
|
||||||
|
|
||||||
%%------------------------------------------------------------------------------
|
%%------------------------------------------------------------------------------
|
||||||
%% Max ClientId Length. Why 1024? NiDongDe!
|
%% Max ClientId Length. Why 1024? NiDongDe!
|
||||||
%%------------------------------------------------------------------------------
|
%%------------------------------------------------------------------------------
|
||||||
|
|
|
@ -225,13 +225,19 @@ case "$1" in
|
||||||
fi
|
fi
|
||||||
if [[ $# -eq 2 ]] && [[ $2 = "list" ]]; then
|
if [[ $# -eq 2 ]] && [[ $2 = "list" ]]; then
|
||||||
$NODETOOL rpc emqttd_ctl bridges list
|
$NODETOOL rpc emqttd_ctl bridges list
|
||||||
elif [ $# -eq 4 ]; then
|
elif [[ $# -eq 2 ]] && [[ $2 = "options" ]]; then
|
||||||
|
$NODETOOL rpc emqttd_ctl bridges options
|
||||||
|
elif [[ $# -eq 4 ]] && [[ $2 = "stop" ]]; then
|
||||||
|
shift
|
||||||
|
$NODETOOL rpc emqttd_ctl bridges $@
|
||||||
|
elif [[ $# -ge 4 ]] && [[ $2 = "start" ]]; then
|
||||||
shift
|
shift
|
||||||
$NODETOOL rpc emqttd_ctl bridges $@
|
$NODETOOL rpc emqttd_ctl bridges $@
|
||||||
else
|
else
|
||||||
echo "Usage: "
|
echo "Usage: "
|
||||||
echo "$SCRIPT bridges list"
|
echo "$SCRIPT bridges list"
|
||||||
echo "$SCRIPT bridges start <Node> <Topic>"
|
echo "$SCRIPT bridges start <Node> <Topic>"
|
||||||
|
echo "$SCRIPT bridges start <Node> <Topic> <Options>"
|
||||||
echo "$SCRIPT bridges stop <Node> <Topic>"
|
echo "$SCRIPT bridges stop <Node> <Topic>"
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
@ -308,8 +314,10 @@ case "$1" in
|
||||||
echo " plugins unload <Plugin> #unload plugin"
|
echo " plugins unload <Plugin> #unload plugin"
|
||||||
echo " ----------------------------------------------------------------"
|
echo " ----------------------------------------------------------------"
|
||||||
echo " bridges list #query bridges"
|
echo " bridges list #query bridges"
|
||||||
|
echo " bridges options #bridge options"
|
||||||
echo " bridges start <Node> <Topic> #start bridge"
|
echo " bridges start <Node> <Topic> #start bridge"
|
||||||
echo " bridges stop <Node> <Topic> #stop bridge"
|
echo " bridges start <Node> <Topic> <Options> #start bridge with options"
|
||||||
|
echo " bridges stop <Node> <Topic> #stop bridge"
|
||||||
echo " ----------------------------------------------------------------"
|
echo " ----------------------------------------------------------------"
|
||||||
echo " useradd <Username> <Password> #add user"
|
echo " useradd <Username> <Password> #add user"
|
||||||
echo " userdel <Username> #delete user"
|
echo " userdel <Username> #delete user"
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
{application, emqttd,
|
{application, emqttd,
|
||||||
[
|
[
|
||||||
{description, "Erlang MQTT Broker"},
|
{description, "Erlang MQTT Broker"},
|
||||||
{vsn, "0.9.2"},
|
{vsn, "0.9.3"},
|
||||||
{modules, []},
|
{modules, []},
|
||||||
{registered, []},
|
{registered, []},
|
||||||
{applications, [kernel,
|
{applications, [kernel,
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
%%%
|
%%%
|
||||||
%%% @end
|
%%% @end
|
||||||
%%%-----------------------------------------------------------------------------
|
%%%-----------------------------------------------------------------------------
|
||||||
|
|
||||||
-module(emqttd_bridge).
|
-module(emqttd_bridge).
|
||||||
|
|
||||||
-author("Feng Lee <feng@emqtt.io>").
|
-author("Feng Lee <feng@emqtt.io>").
|
||||||
|
@ -44,18 +45,18 @@
|
||||||
-define(PING_DOWN_INTERVAL, 1000).
|
-define(PING_DOWN_INTERVAL, 1000).
|
||||||
|
|
||||||
-record(state, {node, subtopic,
|
-record(state, {node, subtopic,
|
||||||
qos,
|
qos = ?QOS_2,
|
||||||
topic_suffix = <<>>,
|
topic_suffix = <<>>,
|
||||||
topic_prefix = <<>>,
|
topic_prefix = <<>>,
|
||||||
mqueue = emqttd_mqueue:mqueue(),
|
mqueue :: emqttd_mqueue:mqueue(),
|
||||||
max_queue_len = 0,
|
max_queue_len = 10000,
|
||||||
ping_down_interval = ?PING_DOWN_INTERVAL,
|
ping_down_interval = ?PING_DOWN_INTERVAL,
|
||||||
status = up}).
|
status = up}).
|
||||||
|
|
||||||
-type option() :: {max_queue_len, pos_integer()} |
|
-type option() :: {qos, mqtt_qos()} |
|
||||||
{qos, mqtt_qos()} |
|
|
||||||
{topic_suffix, binary()} |
|
{topic_suffix, binary()} |
|
||||||
{topic_prefix, binary()} |
|
{topic_prefix, binary()} |
|
||||||
|
{max_queue_len, pos_integer()} |
|
||||||
{ping_down_interval, pos_integer()}.
|
{ping_down_interval, pos_integer()}.
|
||||||
|
|
||||||
-export_type([option/0]).
|
-export_type([option/0]).
|
||||||
|
@ -85,7 +86,7 @@ init([Node, SubTopic, Options]) ->
|
||||||
MQueue = emqttd_mqueue:new(qname(Node, SubTopic),
|
MQueue = emqttd_mqueue:new(qname(Node, SubTopic),
|
||||||
[{max_len, State#state.max_queue_len}],
|
[{max_len, State#state.max_queue_len}],
|
||||||
emqttd_alarm:alarm_fun()),
|
emqttd_alarm:alarm_fun()),
|
||||||
emqttd_pubsub:subscribe({SubTopic, State#state.qos}),
|
emqttd_pubsub:subscribe(SubTopic, State#state.qos),
|
||||||
{ok, State#state{mqueue = MQueue}};
|
{ok, State#state{mqueue = MQueue}};
|
||||||
false ->
|
false ->
|
||||||
{stop, {cannot_connect, Node}}
|
{stop, {cannot_connect, Node}}
|
||||||
|
@ -102,7 +103,9 @@ parse_opts([{topic_prefix, Prefix} | Opts], State) ->
|
||||||
parse_opts([{max_queue_len, Len} | Opts], State) ->
|
parse_opts([{max_queue_len, Len} | Opts], State) ->
|
||||||
parse_opts(Opts, State#state{max_queue_len = Len});
|
parse_opts(Opts, State#state{max_queue_len = Len});
|
||||||
parse_opts([{ping_down_interval, Interval} | Opts], State) ->
|
parse_opts([{ping_down_interval, Interval} | Opts], State) ->
|
||||||
parse_opts(Opts, State#state{ping_down_interval = Interval*1000}).
|
parse_opts(Opts, State#state{ping_down_interval = Interval*1000});
|
||||||
|
parse_opts([_Opt | Opts], State) ->
|
||||||
|
parse_opts(Opts, State).
|
||||||
|
|
||||||
qname(Node, SubTopic) when is_atom(Node) ->
|
qname(Node, SubTopic) when is_atom(Node) ->
|
||||||
qname(atom_to_list(Node), SubTopic);
|
qname(atom_to_list(Node), SubTopic);
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
%%%
|
%%%
|
||||||
%%% @end
|
%%% @end
|
||||||
%%%-----------------------------------------------------------------------------
|
%%%-----------------------------------------------------------------------------
|
||||||
|
|
||||||
-module(emqttd_bridge_sup).
|
-module(emqttd_bridge_sup).
|
||||||
|
|
||||||
-author("Feng Lee <feng@emqtt.io>").
|
-author("Feng Lee <feng@emqtt.io>").
|
||||||
|
@ -63,8 +64,13 @@ start_bridge(Node, SubTopic) when is_atom(Node) and is_binary(SubTopic) ->
|
||||||
|
|
||||||
-spec start_bridge(atom(), binary(), [emqttd_bridge:option()]) -> {ok, pid()} | {error, any()}.
|
-spec start_bridge(atom(), binary(), [emqttd_bridge:option()]) -> {ok, pid()} | {error, any()}.
|
||||||
start_bridge(Node, SubTopic, Options) when is_atom(Node) and is_binary(SubTopic) ->
|
start_bridge(Node, SubTopic, Options) when is_atom(Node) and is_binary(SubTopic) ->
|
||||||
Options1 = emqttd_opts:merge(emqttd_broker:env(bridge), Options),
|
case Node =:= node() of
|
||||||
supervisor:start_child(?MODULE, bridge_spec(Node, SubTopic, Options1)).
|
true ->
|
||||||
|
{error, bridge_to_self};
|
||||||
|
false ->
|
||||||
|
Options1 = emqttd_opts:merge(emqttd_broker:env(bridge), Options),
|
||||||
|
supervisor:start_child(?MODULE, bridge_spec(Node, SubTopic, Options1))
|
||||||
|
end.
|
||||||
|
|
||||||
%%------------------------------------------------------------------------------
|
%%------------------------------------------------------------------------------
|
||||||
%% @doc Stop a bridge
|
%% @doc Stop a bridge
|
||||||
|
|
|
@ -151,12 +151,28 @@ bridges(["list"]) ->
|
||||||
?PRINT("bridge: ~s ~s~n", [Node, Topic])
|
?PRINT("bridge: ~s ~s~n", [Node, Topic])
|
||||||
end, emqttd_bridge_sup:bridges());
|
end, emqttd_bridge_sup:bridges());
|
||||||
|
|
||||||
|
bridges(["options"]) ->
|
||||||
|
?PRINT_MSG("Options:~n"),
|
||||||
|
?PRINT_MSG(" qos = 0 | 1 | 2~n"),
|
||||||
|
?PRINT_MSG(" prefix = string~n"),
|
||||||
|
?PRINT_MSG(" suffix = string~n"),
|
||||||
|
?PRINT_MSG(" queue = integer~n"),
|
||||||
|
?PRINT_MSG("Example:~n"),
|
||||||
|
?PRINT_MSG(" qos=2,prefix=abc/,suffix=/yxz,queue=1000~n");
|
||||||
|
|
||||||
bridges(["start", SNode, Topic]) ->
|
bridges(["start", SNode, Topic]) ->
|
||||||
case emqttd_bridge_sup:start_bridge(list_to_atom(SNode), bin(Topic)) of
|
case emqttd_bridge_sup:start_bridge(list_to_atom(SNode), bin(Topic)) of
|
||||||
{ok, _} -> ?PRINT_MSG("bridge is started.~n");
|
{ok, _} -> ?PRINT_MSG("bridge is started.~n");
|
||||||
{error, Error} -> ?PRINT("error: ~p~n", [Error])
|
{error, Error} -> ?PRINT("error: ~p~n", [Error])
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
bridges(["start", SNode, Topic, OptStr]) ->
|
||||||
|
Opts = parse_opts(bridge, OptStr),
|
||||||
|
case emqttd_bridge_sup:start_bridge(list_to_atom(SNode), bin(Topic), Opts) of
|
||||||
|
{ok, _} -> ?PRINT_MSG("bridge is started.~n");
|
||||||
|
{error, Error} -> ?PRINT("error: ~p~n", [Error])
|
||||||
|
end;
|
||||||
|
|
||||||
bridges(["stop", SNode, Topic]) ->
|
bridges(["stop", SNode, Topic]) ->
|
||||||
case emqttd_bridge_sup:stop_bridge(list_to_atom(SNode), bin(Topic)) of
|
case emqttd_bridge_sup:stop_bridge(list_to_atom(SNode), bin(Topic)) of
|
||||||
ok -> ?PRINT_MSG("bridge is stopped.~n");
|
ok -> ?PRINT_MSG("bridge is stopped.~n");
|
||||||
|
@ -229,3 +245,19 @@ node_name(SNode) ->
|
||||||
bin(S) when is_list(S) -> list_to_binary(S);
|
bin(S) when is_list(S) -> list_to_binary(S);
|
||||||
bin(B) when is_binary(B) -> B.
|
bin(B) when is_binary(B) -> B.
|
||||||
|
|
||||||
|
parse_opts(Cmd, OptStr) ->
|
||||||
|
Tokens = string:tokens(OptStr, ","),
|
||||||
|
[parse_opt(Cmd, list_to_atom(Opt), Val)
|
||||||
|
|| [Opt, Val] <- [string:tokens(S, "=") || S <- Tokens]].
|
||||||
|
|
||||||
|
parse_opt(bridge, qos, Qos) ->
|
||||||
|
{qos, list_to_integer(Qos)};
|
||||||
|
parse_opt(bridge, suffix, Suffix) ->
|
||||||
|
{topic_suffix, list_to_binary(Suffix)};
|
||||||
|
parse_opt(bridge, prefix, Prefix) ->
|
||||||
|
{topic_prefix, list_to_binary(Prefix)};
|
||||||
|
parse_opt(bridge, queue, Len) ->
|
||||||
|
{max_queue_len, list_to_integer(Len)};
|
||||||
|
parse_opt(_Cmd, Opt, _Val) ->
|
||||||
|
?PRINT("Bad Option: ~s~n", [Opt]).
|
||||||
|
|
||||||
|
|
|
@ -55,14 +55,14 @@ make(From, Topic, Payload) ->
|
||||||
|
|
||||||
-spec make(From, Qos, Topic, Payload) -> mqtt_message() when
|
-spec make(From, Qos, Topic, Payload) -> mqtt_message() when
|
||||||
From :: atom() | binary(),
|
From :: atom() | binary(),
|
||||||
Qos :: mqtt_qos(),
|
Qos :: mqtt_qos() | mqtt_qos_name(),
|
||||||
Topic :: binary(),
|
Topic :: binary(),
|
||||||
Payload :: binary().
|
Payload :: binary().
|
||||||
make(From, Qos, Topic, Payload) ->
|
make(From, Qos, Topic, Payload) ->
|
||||||
#mqtt_message{msgid = msgid(Qos),
|
#mqtt_message{msgid = msgid(Qos),
|
||||||
topic = Topic,
|
topic = Topic,
|
||||||
from = From,
|
from = From,
|
||||||
qos = Qos,
|
qos = ?QOS_I(Qos),
|
||||||
payload = Payload,
|
payload = Payload,
|
||||||
timestamp = os:timestamp()}.
|
timestamp = os:timestamp()}.
|
||||||
|
|
||||||
|
@ -107,7 +107,7 @@ from_packet(ClientId, Packet) ->
|
||||||
|
|
||||||
msgid(?QOS_0) ->
|
msgid(?QOS_0) ->
|
||||||
undefined;
|
undefined;
|
||||||
msgid(_Qos) ->
|
msgid(Qos) when Qos =:= ?QOS_1 orelse Qos =:= ?QOS_2 ->
|
||||||
emqttd_guid:gen().
|
emqttd_guid:gen().
|
||||||
|
|
||||||
%%------------------------------------------------------------------------------
|
%%------------------------------------------------------------------------------
|
||||||
|
|
|
@ -43,7 +43,7 @@
|
||||||
-export([start_link/2]).
|
-export([start_link/2]).
|
||||||
|
|
||||||
-export([create/1,
|
-export([create/1,
|
||||||
subscribe/1,
|
subscribe/1, subscribe/2,
|
||||||
unsubscribe/1,
|
unsubscribe/1,
|
||||||
publish/1]).
|
publish/1]).
|
||||||
|
|
||||||
|
@ -131,12 +131,18 @@ create(Topic) when is_binary(Topic) ->
|
||||||
-spec subscribe({Topic, Qos} | list({Topic, Qos})) ->
|
-spec subscribe({Topic, Qos} | list({Topic, Qos})) ->
|
||||||
{ok, Qos | list(Qos)} | {error, any()} when
|
{ok, Qos | list(Qos)} | {error, any()} when
|
||||||
Topic :: binary(),
|
Topic :: binary(),
|
||||||
Qos :: mqtt_qos().
|
Qos :: mqtt_qos() | mqtt_qos_name().
|
||||||
subscribe({Topic, Qos}) when is_binary(Topic) andalso ?IS_QOS(Qos) ->
|
subscribe({Topic, Qos}) when is_binary(Topic) andalso (?IS_QOS(Qos) orelse is_atom(Qos)) ->
|
||||||
call({subscribe, self(), Topic, Qos});
|
call({subscribe, self(), Topic, ?QOS_I(Qos)});
|
||||||
|
|
||||||
subscribe(Topics = [{_Topic, _Qos} | _]) ->
|
subscribe(Topics = [{_Topic, _Qos} | _]) ->
|
||||||
call({subscribe, self(), Topics}).
|
call({subscribe, self(), [{Topic, ?QOS_I(Qos)} || {Topic, Qos} <- Topics]}).
|
||||||
|
|
||||||
|
-spec subscribe(Topic, Qos) -> {ok, Qos} when
|
||||||
|
Topic :: binary(),
|
||||||
|
Qos :: mqtt_qos() | mqtt_qos_name().
|
||||||
|
subscribe(Topic, Qos) ->
|
||||||
|
subscribe({Topic, Qos}).
|
||||||
|
|
||||||
%%------------------------------------------------------------------------------
|
%%------------------------------------------------------------------------------
|
||||||
%% @doc Unsubscribe Topic or Topics
|
%% @doc Unsubscribe Topic or Topics
|
||||||
|
|
|
@ -0,0 +1,47 @@
|
||||||
|
%%%-----------------------------------------------------------------------------
|
||||||
|
%%% Copyright (c) 2012-2015 eMQTT.IO, All Rights Reserved.
|
||||||
|
%%%
|
||||||
|
%%% Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
%%% of this software and associated documentation files (the "Software"), to deal
|
||||||
|
%%% in the Software without restriction, including without limitation the rights
|
||||||
|
%%% to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
%%% copies of the Software, and to permit persons to whom the Software is
|
||||||
|
%%% furnished to do so, subject to the following conditions:
|
||||||
|
%%%
|
||||||
|
%%% The above copyright notice and this permission notice shall be included in all
|
||||||
|
%%% copies or substantial portions of the Software.
|
||||||
|
%%%
|
||||||
|
%%% THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
%%% IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
%%% FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
%%% AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
%%% LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
%%% OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
%%% SOFTWARE.
|
||||||
|
%%%-----------------------------------------------------------------------------
|
||||||
|
%%% @doc
|
||||||
|
%%% emqttd Qos Functions.
|
||||||
|
%%%
|
||||||
|
%%% @end
|
||||||
|
%%%-----------------------------------------------------------------------------
|
||||||
|
-module(emqttd_qos).
|
||||||
|
|
||||||
|
-include("emqttd_protocol.hrl").
|
||||||
|
|
||||||
|
-export([a/1, i/1]).
|
||||||
|
|
||||||
|
a(?QOS_0) -> qos0;
|
||||||
|
a(?QOS_1) -> qos1;
|
||||||
|
a(?QOS_2) -> qos2;
|
||||||
|
a(qos0) -> qos0;
|
||||||
|
a(qos1) -> qos1;
|
||||||
|
a(qos2) -> qos2.
|
||||||
|
|
||||||
|
i(?QOS_0) -> ?QOS_0;
|
||||||
|
i(?QOS_1) -> ?QOS_1;
|
||||||
|
i(?QOS_2) -> ?QOS_2;
|
||||||
|
i(qos0) -> ?QOS_0;
|
||||||
|
i(qos1) -> ?QOS_1;
|
||||||
|
i(qos2) -> ?QOS_2.
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue