|
@@ -38,7 +38,8 @@
|
|
|
'/bridges/:id'/2,
|
|
'/bridges/:id'/2,
|
|
|
'/bridges/:id/:operation'/2,
|
|
'/bridges/:id/:operation'/2,
|
|
|
'/nodes/:node/bridges/:id/:operation'/2,
|
|
'/nodes/:node/bridges/:id/:operation'/2,
|
|
|
- '/bridges/:id/reset_metrics'/2
|
|
|
|
|
|
|
+ '/bridges/:id/metrics'/2,
|
|
|
|
|
+ '/bridges/:id/metrics/reset'/2
|
|
|
]).
|
|
]).
|
|
|
|
|
|
|
|
-export([lookup_from_local_node/2]).
|
|
-export([lookup_from_local_node/2]).
|
|
@@ -68,7 +69,8 @@ paths() ->
|
|
|
"/bridges/:id",
|
|
"/bridges/:id",
|
|
|
"/bridges/:id/:operation",
|
|
"/bridges/:id/:operation",
|
|
|
"/nodes/:node/bridges/:id/:operation",
|
|
"/nodes/:node/bridges/:id/:operation",
|
|
|
- "/bridges/:id/reset_metrics"
|
|
|
|
|
|
|
+ "/bridges/:id/metrics",
|
|
|
|
|
+ "/bridges/:id/metrics/reset"
|
|
|
].
|
|
].
|
|
|
|
|
|
|
|
error_schema(Code, Message) when is_atom(Code) ->
|
|
error_schema(Code, Message) when is_atom(Code) ->
|
|
@@ -132,19 +134,22 @@ param_path_id() ->
|
|
|
}
|
|
}
|
|
|
)}.
|
|
)}.
|
|
|
|
|
|
|
|
-bridge_info_array_example(Method) ->
|
|
|
|
|
- [Config || #{value := Config} <- maps:values(bridge_info_examples(Method))].
|
|
|
|
|
|
|
+bridge_info_array_example(Method, WithMetrics) ->
|
|
|
|
|
+ [Config || #{value := Config} <- maps:values(bridge_info_examples(Method, WithMetrics))].
|
|
|
|
|
|
|
|
bridge_info_examples(Method) ->
|
|
bridge_info_examples(Method) ->
|
|
|
|
|
+ bridge_info_examples(Method, false).
|
|
|
|
|
+
|
|
|
|
|
+bridge_info_examples(Method, WithMetrics) ->
|
|
|
maps:merge(
|
|
maps:merge(
|
|
|
#{
|
|
#{
|
|
|
<<"webhook_example">> => #{
|
|
<<"webhook_example">> => #{
|
|
|
summary => <<"WebHook">>,
|
|
summary => <<"WebHook">>,
|
|
|
- value => info_example(webhook, Method)
|
|
|
|
|
|
|
+ value => info_example(webhook, Method, WithMetrics)
|
|
|
},
|
|
},
|
|
|
<<"mqtt_example">> => #{
|
|
<<"mqtt_example">> => #{
|
|
|
summary => <<"MQTT Bridge">>,
|
|
summary => <<"MQTT Bridge">>,
|
|
|
- value => info_example(mqtt, Method)
|
|
|
|
|
|
|
+ value => info_example(mqtt, Method, WithMetrics)
|
|
|
}
|
|
}
|
|
|
},
|
|
},
|
|
|
ee_bridge_examples(Method)
|
|
ee_bridge_examples(Method)
|
|
@@ -157,24 +162,24 @@ ee_bridge_examples(Method) ->
|
|
|
_:_ -> #{}
|
|
_:_ -> #{}
|
|
|
end.
|
|
end.
|
|
|
|
|
|
|
|
-info_example(Type, Method) ->
|
|
|
|
|
|
|
+info_example(Type, Method, WithMetrics) ->
|
|
|
maps:merge(
|
|
maps:merge(
|
|
|
info_example_basic(Type),
|
|
info_example_basic(Type),
|
|
|
- method_example(Type, Method)
|
|
|
|
|
|
|
+ method_example(Type, Method, WithMetrics)
|
|
|
).
|
|
).
|
|
|
|
|
|
|
|
-method_example(Type, Method) when Method == get; Method == post ->
|
|
|
|
|
|
|
+method_example(Type, Method, WithMetrics) when Method == get; Method == post ->
|
|
|
SType = atom_to_list(Type),
|
|
SType = atom_to_list(Type),
|
|
|
SName = SType ++ "_example",
|
|
SName = SType ++ "_example",
|
|
|
TypeNameExam = #{
|
|
TypeNameExam = #{
|
|
|
type => bin(SType),
|
|
type => bin(SType),
|
|
|
name => bin(SName)
|
|
name => bin(SName)
|
|
|
},
|
|
},
|
|
|
- maybe_with_metrics_example(TypeNameExam, Method);
|
|
|
|
|
-method_example(_Type, put) ->
|
|
|
|
|
|
|
+ maybe_with_metrics_example(TypeNameExam, Method, WithMetrics);
|
|
|
|
|
+method_example(_Type, put, _WithMetrics) ->
|
|
|
#{}.
|
|
#{}.
|
|
|
|
|
|
|
|
-maybe_with_metrics_example(TypeNameExam, get) ->
|
|
|
|
|
|
|
+maybe_with_metrics_example(TypeNameExam, get, true) ->
|
|
|
TypeNameExam#{
|
|
TypeNameExam#{
|
|
|
metrics => ?EMPTY_METRICS,
|
|
metrics => ?EMPTY_METRICS,
|
|
|
node_metrics => [
|
|
node_metrics => [
|
|
@@ -184,7 +189,7 @@ maybe_with_metrics_example(TypeNameExam, get) ->
|
|
|
}
|
|
}
|
|
|
]
|
|
]
|
|
|
};
|
|
};
|
|
|
-maybe_with_metrics_example(TypeNameExam, _) ->
|
|
|
|
|
|
|
+maybe_with_metrics_example(TypeNameExam, _, _) ->
|
|
|
TypeNameExam.
|
|
TypeNameExam.
|
|
|
|
|
|
|
|
info_example_basic(webhook) ->
|
|
info_example_basic(webhook) ->
|
|
@@ -274,7 +279,7 @@ schema("/bridges") ->
|
|
|
responses => #{
|
|
responses => #{
|
|
|
200 => emqx_dashboard_swagger:schema_with_example(
|
|
200 => emqx_dashboard_swagger:schema_with_example(
|
|
|
array(emqx_bridge_schema:get_response()),
|
|
array(emqx_bridge_schema:get_response()),
|
|
|
- bridge_info_array_example(get)
|
|
|
|
|
|
|
+ bridge_info_array_example(get, true)
|
|
|
)
|
|
)
|
|
|
}
|
|
}
|
|
|
},
|
|
},
|
|
@@ -334,9 +339,23 @@ schema("/bridges/:id") ->
|
|
|
}
|
|
}
|
|
|
}
|
|
}
|
|
|
};
|
|
};
|
|
|
-schema("/bridges/:id/reset_metrics") ->
|
|
|
|
|
|
|
+schema("/bridges/:id/metrics") ->
|
|
|
#{
|
|
#{
|
|
|
- 'operationId' => '/bridges/:id/reset_metrics',
|
|
|
|
|
|
|
+ 'operationId' => '/bridges/:id/metrics',
|
|
|
|
|
+ get => #{
|
|
|
|
|
+ tags => [<<"bridges">>],
|
|
|
|
|
+ summary => <<"Get Bridge Metrics">>,
|
|
|
|
|
+ description => ?DESC("desc_bridge_metrics"),
|
|
|
|
|
+ parameters => [param_path_id()],
|
|
|
|
|
+ responses => #{
|
|
|
|
|
+ 200 => emqx_bridge_schema:metrics_fields(),
|
|
|
|
|
+ 404 => error_schema('NOT_FOUND', "Bridge not found")
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+ };
|
|
|
|
|
+schema("/bridges/:id/metrics/reset") ->
|
|
|
|
|
+ #{
|
|
|
|
|
+ 'operationId' => '/bridges/:id/metrics/reset',
|
|
|
put => #{
|
|
put => #{
|
|
|
tags => [<<"bridges">>],
|
|
tags => [<<"bridges">>],
|
|
|
summary => <<"Reset Bridge Metrics">>,
|
|
summary => <<"Reset Bridge Metrics">>,
|
|
@@ -455,7 +474,10 @@ schema("/nodes/:node/bridges/:id/:operation") ->
|
|
|
end
|
|
end
|
|
|
).
|
|
).
|
|
|
|
|
|
|
|
-'/bridges/:id/reset_metrics'(put, #{bindings := #{id := Id}}) ->
|
|
|
|
|
|
|
+'/bridges/:id/metrics'(get, #{bindings := #{id := Id}}) ->
|
|
|
|
|
+ ?TRY_PARSE_ID(Id, lookup_from_all_nodes_metrics(BridgeType, BridgeName, 200)).
|
|
|
|
|
+
|
|
|
|
|
+'/bridges/:id/metrics/reset'(put, #{bindings := #{id := Id}}) ->
|
|
|
?TRY_PARSE_ID(
|
|
?TRY_PARSE_ID(
|
|
|
Id,
|
|
Id,
|
|
|
case
|
|
case
|
|
@@ -469,10 +491,18 @@ schema("/nodes/:node/bridges/:id/:operation") ->
|
|
|
).
|
|
).
|
|
|
|
|
|
|
|
lookup_from_all_nodes(BridgeType, BridgeName, SuccCode) ->
|
|
lookup_from_all_nodes(BridgeType, BridgeName, SuccCode) ->
|
|
|
|
|
+ FormatFun = fun format_bridge_info_without_metrics/1,
|
|
|
|
|
+ do_lookup_from_all_nodes(BridgeType, BridgeName, SuccCode, FormatFun).
|
|
|
|
|
+
|
|
|
|
|
+lookup_from_all_nodes_metrics(BridgeType, BridgeName, SuccCode) ->
|
|
|
|
|
+ FormatFun = fun format_bridge_metrics/1,
|
|
|
|
|
+ do_lookup_from_all_nodes(BridgeType, BridgeName, SuccCode, FormatFun).
|
|
|
|
|
+
|
|
|
|
|
+do_lookup_from_all_nodes(BridgeType, BridgeName, SuccCode, FormatFun) ->
|
|
|
Nodes = mria_mnesia:running_nodes(),
|
|
Nodes = mria_mnesia:running_nodes(),
|
|
|
case is_ok(emqx_bridge_proto_v1:lookup_from_all_nodes(Nodes, BridgeType, BridgeName)) of
|
|
case is_ok(emqx_bridge_proto_v1:lookup_from_all_nodes(Nodes, BridgeType, BridgeName)) of
|
|
|
{ok, [{ok, _} | _] = Results} ->
|
|
{ok, [{ok, _} | _] = Results} ->
|
|
|
- {SuccCode, format_bridge_info([R || {ok, R} <- Results])};
|
|
|
|
|
|
|
+ {SuccCode, FormatFun([R || {ok, R} <- Results])};
|
|
|
{ok, [{error, not_found} | _]} ->
|
|
{ok, [{error, not_found} | _]} ->
|
|
|
{404, error_msg('NOT_FOUND', <<"not_found">>)};
|
|
{404, error_msg('NOT_FOUND', <<"not_found">>)};
|
|
|
{error, ErrL} ->
|
|
{error, ErrL} ->
|
|
@@ -572,7 +602,7 @@ zip_bridges([BridgesFirstNode | _] = BridgesAllNodes) ->
|
|
|
lists:foldl(
|
|
lists:foldl(
|
|
|
fun(#{type := Type, name := Name}, Acc) ->
|
|
fun(#{type := Type, name := Name}, Acc) ->
|
|
|
Bridges = pick_bridges_by_id(Type, Name, BridgesAllNodes),
|
|
Bridges = pick_bridges_by_id(Type, Name, BridgesAllNodes),
|
|
|
- [format_bridge_info(Bridges) | Acc]
|
|
|
|
|
|
|
+ [format_bridge_info_with_metrics(Bridges) | Acc]
|
|
|
end,
|
|
end,
|
|
|
[],
|
|
[],
|
|
|
BridgesFirstNode
|
|
BridgesFirstNode
|
|
@@ -606,7 +636,7 @@ pick_bridges_by_id(Type, Name, BridgesAllNodes) ->
|
|
|
BridgesAllNodes
|
|
BridgesAllNodes
|
|
|
).
|
|
).
|
|
|
|
|
|
|
|
-format_bridge_info([FirstBridge | _] = Bridges) ->
|
|
|
|
|
|
|
+format_bridge_info_with_metrics([FirstBridge | _] = Bridges) ->
|
|
|
Res = maps:remove(node, FirstBridge),
|
|
Res = maps:remove(node, FirstBridge),
|
|
|
NodeStatus = collect_status(Bridges),
|
|
NodeStatus = collect_status(Bridges),
|
|
|
NodeMetrics = collect_metrics(Bridges),
|
|
NodeMetrics = collect_metrics(Bridges),
|
|
@@ -617,6 +647,14 @@ format_bridge_info([FirstBridge | _] = Bridges) ->
|
|
|
node_metrics => NodeMetrics
|
|
node_metrics => NodeMetrics
|
|
|
}).
|
|
}).
|
|
|
|
|
|
|
|
|
|
+format_bridge_info_without_metrics(Bridges) ->
|
|
|
|
|
+ Res = format_bridge_info_with_metrics(Bridges),
|
|
|
|
|
+ maps:without([metrics, node_metrics], Res).
|
|
|
|
|
+
|
|
|
|
|
+format_bridge_metrics(Bridges) ->
|
|
|
|
|
+ Res = format_bridge_info_with_metrics(Bridges),
|
|
|
|
|
+ maps:with([metrics, node_metrics], Res).
|
|
|
|
|
+
|
|
|
collect_status(Bridges) ->
|
|
collect_status(Bridges) ->
|
|
|
[maps:with([node, status], B) || B <- Bridges].
|
|
[maps:with([node, status], B) || B <- Bridges].
|
|
|
|
|
|