Bladeren bron

Merge branch 'emqx30-dev' of github.com:emqtt/emqttd into emqx30-dev

Feng Lee 7 jaren geleden
bovenliggende
commit
694485252a
9 gewijzigde bestanden met toevoegingen van 509 en 31 verwijderingen
  1. 174 25
      etc/emqx.conf
  2. 25 2
      priv/emqx.schema
  3. 254 0
      src/emqx_bridge1.erl
  4. 45 0
      src/emqx_bridge1_sup.erl
  5. 2 1
      src/emqx_client.erl
  6. 3 1
      src/emqx_message.erl
  7. 1 1
      src/emqx_shared_sub.erl
  8. 2 0
      src/emqx_sup.erl
  9. 3 1
      src/emqx_time.erl

+ 174 - 25
etc/emqx.conf

@@ -1504,16 +1504,20 @@ listener.wss.external.ciphers = ECDHE-ECDSA-AES256-GCM-SHA384,ECDHE-RSA-AES256-G
 ## Bridges
 ##--------------------------------------------------------------------
 
-## Bridge Type.
+##--------------------------------------------------------------------
+## Bridges to edge
+##--------------------------------------------------------------------
+## Bridge type.
 ##
-## Value: local | remote
-bridge.name.type = local
+## Value: Enum
+## Example: out | in
+bridge.edge.type = in
 
 ## Bridge address: node name for local bridge, host:port for remote.
 ##
 ## Value: String
 ## Example: emqx@127.0.0.1,  127.0.0.1:1883
-bridge.name.address = emqx@127.0.0.1
+bridge.edge.address = 127.0.0.1:1883
 
 ## Protocol version of the bridge.
 ##
@@ -1521,76 +1525,221 @@ bridge.name.address = emqx@127.0.0.1
 ## - mqtt5
 ## - mqtt4
 ## - mqtt3
-bridge.name.proto_ver = mqtt4
+bridge.edge.proto_ver = mqtt4
 
 ## The ClientId of a remote bridge.
 ##
 ## Value: String
-bridge.name.client_id = bridge:$name
+bridge.edge.client_id = bridge_edge
 
 ## The Clean start flag of a remote bridge.
 ##
 ## Value: boolean
-bridge.name.clean_start = false
+bridge.edge.clean_start = false
 
 ## The username for a remote bridge.
 ##
 ## Value: String
-bridge.name.username = user
+bridge.edge.username = user
 
 ## The password for a remote bridge.
 ##
 ## Value: String
-bridge.name.password = passwd
+bridge.edge.password = passwd
 
 ## Mountpoint of the bridge.
 ##
 ## Value: String
-bridge.name.mountpoint = bridge/$name/
+## bridge.edge.mountpoint = bridge/edge/
+
+## Ping interval of a down bridge.
+##
+## Value: Duration
+## Default: 10 seconds
+bridge.edge.keepalive = 10s
+
+## Subscriptions of the bridge topic.
+##
+## Value: String
+bridge.edge.subscription.1.topic = #
+
+## Subscriptions of the bridge qos.
+##
+## Value: Number
+bridge.edge.subscription.1.qos = 1
+
+## The pending message queue of a bridge.
+##
+## Value: Number
+bridge.edge.max_pending_messages = 10000
+
+## Start type of the bridge.
+##
+## Value: enum
+## manual
+## auto
+bridge.edge.start_type = manual
+
+## Bridge reconnect count.
+##
+## Value: Number
+bridge.edge.reconnect_count = 10
+
+## Bridge reconnect time.
+##
+## Value: Duration
+## Default: 30 seconds
+bridge.edge.reconnect_time = 30s
 
 ## PEM-encoded CA certificates of the bridge.
 ##
 ## Value: File
-bridge.name.cacertfile = cacert.pem
+## bridge.edge.cacertfile = cacert.pem
 
 ## SSL Certfile of the bridge.
 ##
 ## Value: File
-bridge.name.certfile = cert.pem
+## bridge.edge.certfile = cert.pem
 
 ## SSL Keyfile of the bridge.
 ##
 ## Value: File
-bridge.name.keyfile = key.pem
+## bridge.edge.keyfile = key.pem
 
 ## SSL Ciphers used by the bridge.
 ##
 ## Value: String
-bridge.name.ciphers = ECDHE-ECDSA-AES256-GCM-SHA384,ECDHE-RSA-AES256-GCM-SHA384
+## bridge.edge.ciphers = ECDHE-ECDSA-AES256-GCM-SHA384,ECDHE-RSA-AES256-GCM-SHA384
 
 ## TLS versions used by the bridge.
 ##
 ## Value: String
-bridge.name.tls_versions = tlsv1.2,tlsv1.1,tlsv1
+## bridge.edge.tls_versions = tlsv1.2,tlsv1.1,tlsv1
 
-## The pending message queue of a bridge.
+
+##--------------------------------------------------------------------
+## Bridges to cloud
+##--------------------------------------------------------------------
+## Bridge type.
 ##
-## Value: Number
-bridge.name.max_pending_messages = 10000
+## Value: Enum
+## Example: out | in
+bridge.cloud.type = out
+
+## Bridge address: node name for local bridge, host:port for remote.
+##
+## Value: String
+## Example: emqx@127.0.0.1,  127.0.0.1:1883
+bridge.cloud.address = 127.0.0.1:1883
+
+## Protocol version of the bridge.
+##
+## Value: Enum
+## - mqtt5
+## - mqtt4
+## - mqtt3
+bridge.cloud.proto_ver = mqtt4
+
+## The ClientId of a remote bridge.
+##
+## Value: String
+bridge.cloud.client_id = bridge_cloud
+
+## The Clean start flag of a remote bridge.
+##
+## Value: boolean
+bridge.cloud.clean_start = false
+
+## The username for a remote bridge.
+##
+## Value: String
+bridge.cloud.username = user
+
+## The password for a remote bridge.
+##
+## Value: String
+bridge.cloud.password = passwd
+
+## Mountpoint of the bridge.
+##
+## Value: String
+bridge.cloud.mountpoint = bridge/edge/${node}/
 
 ## Ping interval of a down bridge.
 ##
 ## Value: Duration
 ## Default: 10 seconds
-bridge.name.keepalive = 10s
+bridge.cloud.keepalive = 10s
 
-## Subscriptions of the bridge.
+## Forward message topics
 ##
-## Default: 10 seconds
-bridge.name.subscription.1.topic = topic1/
-bridge.name.subscription.1.qos = 2
-## bridge.name.subscription.2.topic = topic2/
-## bridge.name.subscription.2.qos = 2
+## Value: String
+## Example: topic1/#,topic2/#
+bridge.cloud.forward_rule = #
+
+## Subscriptions of the bridge topic.
+##
+## Value: String
+bridge.cloud.subscription.1.topic = $share/cmd/topic1
+
+## Subscriptions of the bridge qos.
+##
+## Value: Number
+bridge.cloud.subscription.1.qos = 1
+
+## Bridge store message type.
+##
+## Value: Enum
+## Example: memory | disk
+bridge.cloud.store_type = memory
+
+## The pending message queue of a bridge.
+##
+## Value: Number
+bridge.cloud.max_pending_messages = 10000
+
+## Start type of the bridge.
+##
+## Value: enum
+## manual
+## auto
+bridge.cloud.start_type = manual
+
+## Bridge reconnect count.
+##
+## Value: Number
+bridge.cloud.reconnect_count = 10
+
+## Bridge reconnect time.
+##
+## Value: Duration
+## Default: 30 seconds
+bridge.cloud.reconnect_time = 30s
+
+## PEM-encoded CA certificates of the bridge.
+##
+## Value: File
+## bridge.cloud.cacertfile = cacert.pem
+
+## SSL Certfile of the bridge.
+##
+## Value: File
+## bridge.cloud.certfile = cert.pem
+
+## SSL Keyfile of the bridge.
+##
+## Value: File
+## bridge.cloud.keyfile = key.pem
+
+## SSL Ciphers used by the bridge.
+##
+## Value: String
+## bridge.cloud.ciphers = ECDHE-ECDSA-AES256-GCM-SHA384,ECDHE-RSA-AES256-GCM-SHA384
+
+## TLS versions used by the bridge.
+##
+## Value: String
+## bridge.cloud.tls_versions = tlsv1.2,tlsv1.1,tlsv1
 
 ##--------------------------------------------------------------------
 ## Modules

+ 25 - 2
priv/emqx.schema

@@ -1411,8 +1411,11 @@ end}.
 %%--------------------------------------------------------------------
 
 {mapping, "bridge.$name.type", "emqx.bridges", [
-  {default, local},
-  {datatype, {enum, [local,remote]}}
+  {datatype, {enum, [in, out]}}
+]}.
+
+{mapping, "bridge.$name.store_type", "emqx.bridges", [
+  {datatype, {enum, [memory, disk]}}
 ]}.
 
 {mapping, "bridge.$name.address", "emqx.bridges", [
@@ -1444,6 +1447,10 @@ end}.
   {datatype, string}
 ]}.
 
+{mapping, "bridge.$name.forward_rule", "emqx.bridges", [
+  {datatype, string}
+]}.
+
 {mapping, "bridge.$name.cacertfile", "emqx.bridges", [
   {datatype, string}
 ]}.
@@ -1482,6 +1489,22 @@ end}.
   {datatype, integer}
 ]}.
 
+{mapping, "bridge.$name.start_type", "emqx.bridges", [
+  {datatype, {enum, [manual, auto]}},
+  {default, auto}
+]}.
+
+{mapping, "bridge.$name.reconnect_count", "emqx.bridges", [
+  {default, 10},
+  {datatype, integer}
+]}.
+
+{mapping, "bridge.$name.reconnect_time", "emqx.bridges", [
+  {default, "30s"},
+  {datatype, {duration, s}}
+]}.
+
+
 {translation, "emqx.bridges", fun(Conf) ->
 
     Split = fun(undefined) -> undefined; (S) -> string:tokens(S, ",") end,

+ 254 - 0
src/emqx_bridge1.erl

@@ -0,0 +1,254 @@
+%% Copyright (c) 2018 EMQ Technologies Co., Ltd. All Rights Reserved.
+%%
+%% Licensed under the Apache License, Version 2.0 (the "License");
+%% you may not use this file except in compliance with the License.
+%% You may obtain a copy of the License at
+%%
+%%     http://www.apache.org/licenses/LICENSE-2.0
+%%
+%% Unless required by applicable law or agreed to in writing, software
+%% distributed under the License is distributed on an "AS IS" BASIS,
+%% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+%% See the License for the specific language governing permissions and
+%% limitations under the License.
+
+-module(emqx_bridge1).
+
+-behaviour(gen_server).
+
+-include("emqx.hrl").
+-include("emqx_mqtt.hrl").
+
+-import(proplists, [get_value/2, get_value/3]).
+
+-export([start_link/2, start_bridge/1, stop_bridge/1, status/1]).
+
+-export([init/1, handle_call/3, handle_cast/2, handle_info/2, terminate/2,
+         code_change/3]).
+
+-record(state, {client_pid, options, reconnect_time, reconnect_count,
+                def_reconnect_count, type, mountpoint, queue, store_type,
+                max_pending_messages}).
+
+-record(mqtt_msg, {qos = ?QOS0, retain = false, dup = false,
+                   packet_id, topic, props, payload}).
+
+start_link(Name, Options) ->
+    gen_server:start_link({local, name(Name)}, ?MODULE, [Options], []).
+
+start_bridge(Name) ->
+    gen_server:call(name(Name), start_bridge).
+
+stop_bridge(Name) ->
+    gen_server:call(name(Name), stop_bridge).
+
+status(Pid) ->
+    gen_server:call(Pid, status).
+
+%%------------------------------------------------------------------------------
+%% gen_server callbacks
+%%------------------------------------------------------------------------------
+
+init([Options]) ->
+    process_flag(trap_exit, true),
+    case get_value(start_type, Options, manual) of
+        manual -> ok;
+        auto -> erlang:send_after(1000, self(), start)
+    end,
+    ReconnectCount = get_value(reconnect_count, Options, 10),
+    ReconnectTime = get_value(reconnect_time, Options, 30000),
+    MaxPendingMsg = get_value(max_pending_messages, Options, 10000),
+    Mountpoint = format_mountpoint(get_value(mountpoint, Options)),
+    StoreType = get_value(store_type, Options, memory),
+    Type = get_value(type, Options, in),
+    Queue = [],
+    {ok, #state{type                = Type,
+                mountpoint          = Mountpoint,
+                queue               = Queue,
+                store_type          = StoreType,
+                options             = Options,
+                reconnect_count     = ReconnectCount,
+                reconnect_time      = ReconnectTime,
+                def_reconnect_count = ReconnectCount,
+                max_pending_messages = MaxPendingMsg}}.
+
+handle_call(start_bridge, _From, State = #state{client_pid = undefined}) ->
+    {noreply, NewState} = handle_info(start, State),
+    {reply, <<"start bridge successfully">>, NewState};
+
+handle_call(start_bridge, _From, State) ->
+    {reply, <<"bridge already started">>, State};
+
+handle_call(stop_bridge, _From, State = #state{client_pid = undefined}) ->
+    {reply, <<"bridge not started">>, State};
+
+handle_call(stop_bridge, _From, State = #state{client_pid = Pid}) ->
+    emqx_client:disconnect(Pid),
+    {reply, <<"stop bridge successfully">>, State};
+
+handle_call(status, _From, State = #state{client_pid = undefined}) ->
+    {reply, <<"Stopped">>, State};
+handle_call(status, _From, State = #state{client_pid = _Pid})->
+    {reply, <<"Running">>, State};
+
+handle_call(Req, _From, State) ->
+    emqx_logger:error("[Bridge] unexpected call: ~p", [Req]),
+    {reply, ignored, State}.
+
+handle_cast(Msg, State) ->
+    emqx_logger:error("[Bridge] unexpected cast: ~p", [Msg]),
+    {noreply, State}.
+
+handle_info(start, State = #state{reconnect_count = 0}) ->
+    {noreply, State};
+
+%%----------------------------------------------------------------
+%% start in message bridge
+%%----------------------------------------------------------------
+handle_info(start, State = #state{options = Options,
+                                  client_pid = undefined,
+                                  reconnect_time = ReconnectTime,
+                                  reconnect_count = ReconnectCount,
+                                  type = in}) ->
+    case emqx_client:start_link([{owner, self()}|options(Options)]) of
+        {ok, ClientPid, _} ->
+            Subs = get_value(subscriptions, Options, []),
+            [emqx_client:subscribe(ClientPid, {i2b(Topic), Qos}) || {Topic, Qos} <- Subs],
+            {noreply, State#state{client_pid = ClientPid}};
+        {error,_} ->
+            erlang:send_after(ReconnectTime, self(), start),
+            {noreply, State = #state{reconnect_count = ReconnectCount-1}}
+    end;
+
+%%----------------------------------------------------------------
+%% start out message bridge
+%%----------------------------------------------------------------
+handle_info(start, State = #state{options = Options,
+                                  client_pid = undefined,
+                                  reconnect_time = ReconnectTime,
+                                  reconnect_count = ReconnectCount,
+                                  type = out}) ->
+    case emqx_client:start_link([{owner, self()}|options(Options)]) of
+        {ok, ClientPid, _} ->
+            Subs = get_value(subscriptions, Options, []),
+            [emqx_client:subscribe(ClientPid, {i2b(Topic), Qos}) || {Topic, Qos} <- Subs],
+            ForwardRules = string:tokens(get_value(forward_rule, Options, ""), ","),
+            [emqx_broker:subscribe(i2b(Topic)) || Topic <- ForwardRules, emqx_topic:validate({filter, i2b(Topic)})],
+            {noreply, State#state{client_pid = ClientPid}};
+        {error,_} ->
+            erlang:send_after(ReconnectTime, self(), start),
+            {noreply, State = #state{reconnect_count = ReconnectCount-1}}
+    end;
+
+%%----------------------------------------------------------------
+%% received local node message
+%%----------------------------------------------------------------
+handle_info({dispatch, _, #message{topic = Topic, payload = Payload, flags = #{retain := Retain}}},
+             State = #state{client_pid = Pid, mountpoint = Mountpoint, queue = Queue,
+                            store_type = StoreType, max_pending_messages = MaxPendingMsg}) ->
+    Msg = #mqtt_msg{qos     = 1,
+                    retain  = Retain,
+                    topic   = mountpoint(Mountpoint, Topic),
+                    payload = Payload},
+    case emqx_client:publish(Pid, Msg) of
+        {ok, PkgId} ->
+            {noreply, State#state{queue = store(StoreType, {PkgId, Msg}, Queue, MaxPendingMsg)}};
+        {error, Reason} ->
+            emqx_logger:error("Publish fail:~p", [Reason]),
+            {noreply, State}
+    end;
+
+%%----------------------------------------------------------------
+%% received remote node message
+%%----------------------------------------------------------------
+handle_info({publish, #{qos := QoS, dup := Dup, retain := Retain, topic := Topic,
+                        properties := Props, payload := Payload}}, State) ->
+    NewMsg0 = emqx_message:make(bridge, QoS, Topic, Payload),
+    NewMsg1 = emqx_message:set_headers(Props, emqx_message:set_flags(#{dup => Dup, retain=> Retain}, NewMsg0)),
+    emqx_broker:publish(NewMsg1),
+    {noreply, State};
+
+%%----------------------------------------------------------------
+%% received remote puback message
+%%----------------------------------------------------------------
+handle_info({puback, #{packet_id := PkgId}}, State = #state{queue = Queue, store_type = StoreType}) ->
+    % lists:keydelete(PkgId, 1, Queue)
+    {noreply, State#state{queue = delete(StoreType, PkgId, Queue)}};
+
+handle_info({'EXIT', Pid, normal}, State = #state{client_pid = Pid}) ->
+    {noreply, State#state{client_pid = undefined}};
+
+handle_info({'EXIT', Pid, Reason}, State = #state{client_pid = Pid,
+                                                  reconnect_time = ReconnectTime,
+                                                  def_reconnect_count = DefReconnectCount}) ->
+    lager:warning("emqx bridge stop reason:~p", [Reason]),
+    erlang:send_after(ReconnectTime, self(), start),
+    {noreply, State#state{client_pid = undefined, reconnect_count = DefReconnectCount}};
+
+handle_info(Info, State) ->
+    emqx_logger:error("[Bridge] unexpected info: ~p", [Info]),
+    {noreply, State}.
+
+terminate(_Reason, #state{}) ->
+    ok.
+
+code_change(_OldVsn, State, _Extra) ->
+    {ok, State}.
+
+proto_ver(mqtt3) -> v3;
+proto_ver(mqtt4) -> v4;
+proto_ver(mqtt5) -> v5.
+address(Address) ->
+    case string:tokens(Address, ":") of
+        [Host] -> {Host, 1883};
+        [Host, Port] -> {Host, list_to_integer(Port)}
+    end.
+options(Options) ->
+    options(Options, []).
+options([], Acc) ->
+    Acc;
+options([{username, Username}| Options], Acc) ->
+    options(Options, [{username, Username}|Acc]);
+options([{proto_ver, ProtoVer}| Options], Acc) ->
+    options(Options, [{proto_ver, proto_ver(ProtoVer)}|Acc]);
+options([{password, Password}| Options], Acc) ->
+    options(Options, [{password, Password}|Acc]);
+options([{keepalive, Keepalive}| Options], Acc) ->
+    options(Options, [{keepalive, Keepalive}|Acc]);
+options([{client_id, ClientId}| Options], Acc) ->
+    options(Options, [{client_id, ClientId}|Acc]);
+options([{clean_start, CleanStart}| Options], Acc) ->
+    options(Options, [{clean_start, CleanStart}|Acc]);
+options([{address, Address}| Options], Acc) ->
+    {Host, Port} = address(Address),
+    options(Options, [{host, Host}, {port, Port}|Acc]);
+options([_Option | Options], Acc) ->
+    options(Options, Acc).
+
+name(Id) ->
+    list_to_atom(lists:concat([?MODULE, "_", Id])).
+
+i2b(L) -> iolist_to_binary(L).
+
+mountpoint(undefined, Topic) ->
+    Topic;
+mountpoint(Prefix, Topic) ->
+    <<Prefix/binary, Topic/binary>>.
+
+format_mountpoint(undefined) ->
+    undefined;
+format_mountpoint(Prefix) ->
+    binary:replace(i2b(Prefix), <<"${node}">>, atom_to_binary(node(), utf8)).
+
+store(memory, Data, Queue, MaxPendingMsg) when length(Queue) =< MaxPendingMsg ->
+    [Data | Queue];
+store(memory, _Data, Queue, _MaxPendingMsg) ->
+    lager:error("Beyond max pending messages"),
+    Queue;
+store(disk, Data, Queue, _MaxPendingMsg)->
+    [Data | Queue].
+
+delete(memory, PkgId, Queue) ->
+    lists:keydelete(PkgId, 1, Queue);
+delete(disk, PkgId, Queue) ->
+    lists:keydelete(PkgId, 1, Queue).

+ 45 - 0
src/emqx_bridge1_sup.erl

@@ -0,0 +1,45 @@
+%% Copyright (c) 2018 EMQ Technologies Co., Ltd. All Rights Reserved.
+%%
+%% Licensed under the Apache License, Version 2.0 (the "License");
+%% you may not use this file except in compliance with the License.
+%% You may obtain a copy of the License at
+%%
+%%     http://www.apache.org/licenses/LICENSE-2.0
+%%
+%% Unless required by applicable law or agreed to in writing, software
+%% distributed under the License is distributed on an "AS IS" BASIS,
+%% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+%% See the License for the specific language governing permissions and
+%% limitations under the License.
+
+-module(emqx_bridge1_sup).
+
+-behavior(supervisor).
+
+-include("emqx.hrl").
+
+-export([start_link/0, bridges/0]).
+
+%% Supervisor callbacks
+-export([init/1]).
+
+start_link() ->
+    supervisor:start_link({local, ?MODULE}, ?MODULE, []).
+
+%% @doc List all bridges
+-spec(bridges() -> [{node(), topic(), pid()}]).
+bridges() ->
+    [{Name, emqx_bridge1:status(Pid)} || {Name, Pid, _, _} <- supervisor:which_children(?MODULE)].
+
+init([]) ->
+    BridgesOpts = emqx_config:get_env(bridges, []),
+    Bridges = [spec(Opts)|| Opts <- BridgesOpts],
+    {ok, {{one_for_one, 10, 100}, Bridges}}.
+
+spec({Id, Options})->
+    #{id       => Id,
+      start    => {emqx_bridge1, start_link, [Id, Options]},
+      restart  => permanent,
+      shutdown => 5000,
+      type     => worker,
+      modules  => [emqx_bridge1]}.

+ 2 - 1
src/emqx_client.erl

@@ -989,7 +989,8 @@ deliver(#mqtt_msg{qos = QoS, dup = Dup, retain = Retain, packet_id = PacketId,
                   topic = Topic, props = Props, payload = Payload},
         State = #state{owner = Owner}) ->
     Owner ! {publish, #{qos => QoS, dup => Dup, retain => Retain, packet_id => PacketId,
-                        topic => Topic, properties => Props, payload => Payload}},
+                        topic => Topic, properties => Props, payload => Payload,
+                        client_pid => self()}},
     State.
 
 packet_to_msg(#mqtt_packet{header   = #mqtt_packet_header{type   = ?PUBLISH,

+ 3 - 1
src/emqx_message.erl

@@ -69,7 +69,9 @@ unset_flag(Flag, Msg = #message{flags = Flags}) ->
 set_headers(Headers, Msg = #message{headers = undefined}) when is_map(Headers) ->
     Msg#message{headers = Headers};
 set_headers(New, Msg = #message{headers = Old}) when is_map(New) ->
-    Msg#message{headers = maps:merge(Old, New)}.
+    Msg#message{headers = maps:merge(Old, New)};
+set_headers(_, Msg) ->
+    Msg.
 
 get_header(Hdr, Msg) ->
     get_header(Hdr, Msg, undefined).

+ 1 - 1
src/emqx_shared_sub.erl

@@ -97,7 +97,7 @@ pick(SubPids) ->
     lists:nth((X rem length(SubPids)) + 1, SubPids).
 
 subscribers(Group, Topic) ->
-    ets:select(?TAB, [{{shared_subscription, Group, Topic, '$1'}, [], ['$1']}]).
+    ets:select(?TAB, [{{emqx_shared_subscription, Group, Topic, '$1'}, [], ['$1']}]).
 
 %%-----------------------------------------------------------------------------
 %% gen_server callbacks

+ 2 - 0
src/emqx_sup.erl

@@ -63,6 +63,7 @@ init([]) ->
     BrokerSup = supervisor_spec(emqx_broker_sup),
     %% BridgeSup
     BridgeSup = supervisor_spec(emqx_bridge_sup_sup),
+    BridgeSup1 = supervisor_spec(emqx_bridge1_sup),
     %% AccessControl
     AccessControl = worker_spec(emqx_access_control),
     %% Session Manager
@@ -78,6 +79,7 @@ init([]) ->
            RouterSup,
            BrokerSup,
            BridgeSup,
+           BridgeSup1,
            AccessControl,
            SMSup,
            SessionSup,

+ 3 - 1
src/emqx_time.erl

@@ -14,7 +14,7 @@
 
 -module(emqx_time).
 
--export([seed/0, now_secs/0, now_ms/0]).
+-export([seed/0, now_secs/0, now_ms/0, now_ms/1]).
 
 seed() ->
     rand:seed(exsplus, erlang:timestamp()).
@@ -25,3 +25,5 @@ now_secs() ->
 now_ms() ->
     erlang:system_time(millisecond).
 
+now_ms({MegaSecs, Secs, MicroSecs}) ->
+     (MegaSecs * 1000000 + Secs) * 1000 + round(MicroSecs/1000).