Version support for time_compat
References rabbitmq/rabbitmq-server#346. References rabbitmq/rabbitmq-server#347.
This commit is contained in:
parent
adbdab6a5d
commit
555e352a4b
|
|
@ -0,0 +1,184 @@
|
||||||
|
%% The contents of this file are subject to the Mozilla Public License
|
||||||
|
%% Version 1.1 (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.mozilla.org/MPL/
|
||||||
|
%%
|
||||||
|
%% Software distributed under the License is distributed on an "AS IS"
|
||||||
|
%% basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See
|
||||||
|
%% the License for the specific language governing rights and
|
||||||
|
%% limitations under the License.
|
||||||
|
%%
|
||||||
|
%% The Original Code is RabbitMQ Federation.
|
||||||
|
%%
|
||||||
|
%% The Initial Developer of the Original Code is GoPivotal, Inc.
|
||||||
|
%% Copyright (c) 2007-2016 Pivotal Software, Inc. All rights reserved.
|
||||||
|
%%
|
||||||
|
-module(code_version).
|
||||||
|
|
||||||
|
-export([update/1]).
|
||||||
|
|
||||||
|
update(Module) ->
|
||||||
|
AbsCode = get_abs_code(Module),
|
||||||
|
Forms = replace_forms(Module, get_otp_version() >= 18, AbsCode),
|
||||||
|
Code = compile_forms(Forms),
|
||||||
|
load_code(Module, Code).
|
||||||
|
|
||||||
|
load_code(Module, Code) ->
|
||||||
|
unload(Module),
|
||||||
|
case code:load_binary(Module, "loaded by rabbit_common", Code) of
|
||||||
|
{module, _} ->
|
||||||
|
ok;
|
||||||
|
{error, _Reason} ->
|
||||||
|
throw(cannot_load)
|
||||||
|
end.
|
||||||
|
|
||||||
|
unload(Module) ->
|
||||||
|
code:soft_purge(Module),
|
||||||
|
code:delete(Module).
|
||||||
|
|
||||||
|
compile_forms(Forms) ->
|
||||||
|
case compile:forms(Forms, [debug_info]) of
|
||||||
|
{ok, _ModName, Code} ->
|
||||||
|
Code;
|
||||||
|
{ok, _ModName, Code, _Warnings} ->
|
||||||
|
Code;
|
||||||
|
_ ->
|
||||||
|
throw(cannot_compile_forms)
|
||||||
|
end.
|
||||||
|
|
||||||
|
get_abs_code(Module) ->
|
||||||
|
get_forms(get_object_code(Module)).
|
||||||
|
|
||||||
|
get_object_code(Module) ->
|
||||||
|
case code:get_object_code(Module) of
|
||||||
|
{_Mod, Code, _File} ->
|
||||||
|
Code;
|
||||||
|
error ->
|
||||||
|
throw(not_found)
|
||||||
|
end.
|
||||||
|
|
||||||
|
get_forms(Code) ->
|
||||||
|
case beam_lib:chunks(Code, [abstract_code]) of
|
||||||
|
{ok, {_, [{abstract_code, {raw_abstract_v1, Forms}}]}} ->
|
||||||
|
Forms;
|
||||||
|
{ok, {_, [{abstract_code, no_abstract_code}]}} ->
|
||||||
|
throw(no_abstract_code);
|
||||||
|
{error, beam_lib, Reason} ->
|
||||||
|
throw({no_abstract_code, Reason})
|
||||||
|
end.
|
||||||
|
|
||||||
|
get_otp_version() ->
|
||||||
|
Version = erlang:system_info(otp_release),
|
||||||
|
case re:run(Version, "^[0-9][0-9]", [{capture, first, list}]) of
|
||||||
|
{match, [V]} ->
|
||||||
|
list_to_integer(V);
|
||||||
|
_ ->
|
||||||
|
%% Could be anything below R17, we are not interested
|
||||||
|
0
|
||||||
|
end.
|
||||||
|
|
||||||
|
get_original_pairs(VersionSupport) ->
|
||||||
|
[{Orig, Arity} || {Orig, _Pre, _Post, Arity} <- VersionSupport].
|
||||||
|
|
||||||
|
get_delete_pairs(true, VersionSupport) ->
|
||||||
|
[{Pre, Arity} || {_Orig, Pre, _Post, Arity} <- VersionSupport];
|
||||||
|
get_delete_pairs(false, VersionSupport) ->
|
||||||
|
[{Post, Arity} || {_Orig, _Pre, Post, Arity} <- VersionSupport].
|
||||||
|
|
||||||
|
get_rename_pairs(true, VersionSupport) ->
|
||||||
|
[{Post, Arity} || {_Orig, _Pre, Post, Arity} <- VersionSupport];
|
||||||
|
get_rename_pairs(false, VersionSupport) ->
|
||||||
|
[{Pre, Arity} || {_Orig, Pre, _Post, Arity} <- VersionSupport].
|
||||||
|
|
||||||
|
get_name_pairs(true, VersionSupport) ->
|
||||||
|
[{Post, Orig} || {Orig, _Pre, Post, _Arity} <- VersionSupport];
|
||||||
|
get_name_pairs(false, VersionSupport) ->
|
||||||
|
[{Pre, Orig} || {Orig, Pre, _Post, _Arity} <- VersionSupport].
|
||||||
|
|
||||||
|
delete_abstract_functions(ToDelete) ->
|
||||||
|
fun(Tree, Function) ->
|
||||||
|
case lists:member(Function, ToDelete) of
|
||||||
|
true ->
|
||||||
|
erl_syntax:comment(["Deleted unused function"]);
|
||||||
|
false ->
|
||||||
|
Tree
|
||||||
|
end
|
||||||
|
end.
|
||||||
|
|
||||||
|
rename_abstract_functions(ToRename, ToName) ->
|
||||||
|
fun(Tree, {Name, _Arity} = Function) ->
|
||||||
|
case lists:member(Function, ToRename) of
|
||||||
|
true ->
|
||||||
|
FunctionName = proplists:get_value(Name, ToName),
|
||||||
|
erl_syntax:function(
|
||||||
|
erl_syntax:atom(FunctionName),
|
||||||
|
erl_syntax:function_clauses(Tree));
|
||||||
|
false ->
|
||||||
|
Tree
|
||||||
|
end
|
||||||
|
end.
|
||||||
|
|
||||||
|
replace_forms(Module, IsPost18, AbsCode) ->
|
||||||
|
Attr = Module:module_info(attributes),
|
||||||
|
VersionSupport = proplists:get_value(version_support, Attr),
|
||||||
|
Original = get_original_pairs(VersionSupport),
|
||||||
|
ToDelete = get_delete_pairs(IsPost18, VersionSupport),
|
||||||
|
DeleteFun = delete_abstract_functions(ToDelete ++ Original),
|
||||||
|
AbsCode0 = replace_function_forms(AbsCode, DeleteFun),
|
||||||
|
ToRename = get_rename_pairs(IsPost18, VersionSupport),
|
||||||
|
ToName = get_name_pairs(IsPost18, VersionSupport),
|
||||||
|
RenameFun = rename_abstract_functions(ToRename, ToName),
|
||||||
|
remove_exports(replace_function_forms(AbsCode0, RenameFun), ToDelete ++ ToRename).
|
||||||
|
|
||||||
|
replace_function_forms(AbsCode, Fun) ->
|
||||||
|
ReplaceFunction =
|
||||||
|
fun(Tree) ->
|
||||||
|
case erl_syntax_lib:analyze_function(Tree) of
|
||||||
|
{_N, _A} = Function ->
|
||||||
|
Fun(Tree, Function);
|
||||||
|
_Other -> Tree
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
Filter = fun(Tree) ->
|
||||||
|
case erl_syntax:type(Tree) of
|
||||||
|
function -> ReplaceFunction(Tree);
|
||||||
|
_Other -> Tree
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
fold_syntax_tree(Filter, AbsCode).
|
||||||
|
|
||||||
|
filter_export_pairs(Info, ToDelete) ->
|
||||||
|
lists:filter(fun(Pair) ->
|
||||||
|
not lists:member(Pair, ToDelete)
|
||||||
|
end, Info).
|
||||||
|
|
||||||
|
remove_exports(AbsCode, ToDelete) ->
|
||||||
|
RemoveExports =
|
||||||
|
fun(Tree) ->
|
||||||
|
case erl_syntax_lib:analyze_attribute(Tree) of
|
||||||
|
{export, Info} ->
|
||||||
|
Remaining = filter_export_pairs(Info, ToDelete),
|
||||||
|
rebuild_export(Remaining);
|
||||||
|
_Other -> Tree
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
Filter = fun(Tree) ->
|
||||||
|
case erl_syntax:type(Tree) of
|
||||||
|
attribute -> RemoveExports(Tree);
|
||||||
|
_Other -> Tree
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
fold_syntax_tree(Filter, AbsCode).
|
||||||
|
|
||||||
|
rebuild_export(Args) ->
|
||||||
|
erl_syntax:attribute(
|
||||||
|
erl_syntax:atom(export),
|
||||||
|
[erl_syntax:list(
|
||||||
|
[erl_syntax:arity_qualifier(erl_syntax:atom(N),
|
||||||
|
erl_syntax:integer(A))
|
||||||
|
|| {N, A} <- Args])]).
|
||||||
|
|
||||||
|
fold_syntax_tree(Filter, Forms) ->
|
||||||
|
Tree = erl_syntax:form_list(Forms),
|
||||||
|
NewTree = erl_syntax_lib:map(Filter, Tree),
|
||||||
|
erl_syntax:revert_forms(NewTree).
|
||||||
|
|
@ -43,110 +43,176 @@
|
||||||
%% where it has not yet been deprecated.
|
%% where it has not yet been deprecated.
|
||||||
%%
|
%%
|
||||||
|
|
||||||
|
-version_support(
|
||||||
|
[{monotonic_time, monotonic_time_pre_18, monotonic_time_post_18, 0},
|
||||||
|
{monotonic_time, monotonic_time_pre_18, monotonic_time_post_18, 1},
|
||||||
|
{erlang_system_time, erlang_system_time_pre_18, erlang_system_time_post_18, 0},
|
||||||
|
{erlang_system_time, erlang_system_time_pre_18, erlang_system_time_post_18, 1},
|
||||||
|
{os_system_time, os_system_time_pre_18, os_system_time_post_18, 0},
|
||||||
|
{os_system_time, os_system_time_pre_18, os_system_time_post_18, 1},
|
||||||
|
{time_offset, time_offset_pre_18, time_offset_post_18, 0},
|
||||||
|
{time_offset, time_offset_pre_18, time_offset_post_18, 1},
|
||||||
|
{convert_time_unit, convert_time_unit_pre_18, convert_time_unit_post_18, 0},
|
||||||
|
{timestamp, timestamp_pre_18, timestamp_post_18, 0},
|
||||||
|
{unique_integer, unique_integer_pre_18, unique_integer_post_18, 0},
|
||||||
|
{unique_integer, unique_integer_pre_18, unique_integer_post_18, 1}]).
|
||||||
|
|
||||||
-export([monotonic_time/0,
|
-export([monotonic_time/0,
|
||||||
|
monotonic_time_pre_18/0,
|
||||||
|
monotonic_time_post_18/0,
|
||||||
monotonic_time/1,
|
monotonic_time/1,
|
||||||
|
monotonic_time_pre_18/1,
|
||||||
|
monotonic_time_post_18/1,
|
||||||
erlang_system_time/0,
|
erlang_system_time/0,
|
||||||
|
erlang_system_time_pre_18/0,
|
||||||
|
erlang_system_time_post_18/0,
|
||||||
erlang_system_time/1,
|
erlang_system_time/1,
|
||||||
|
erlang_system_time_pre_18/1,
|
||||||
|
erlang_system_time_post_18/1,
|
||||||
os_system_time/0,
|
os_system_time/0,
|
||||||
|
os_system_time_pre_18/0,
|
||||||
|
os_system_time_post_18/0,
|
||||||
os_system_time/1,
|
os_system_time/1,
|
||||||
|
os_system_time_pre_18/1,
|
||||||
|
os_system_time_post_18/1,
|
||||||
time_offset/0,
|
time_offset/0,
|
||||||
|
time_offset_pre_18/0,
|
||||||
|
time_offset_post_18/0,
|
||||||
time_offset/1,
|
time_offset/1,
|
||||||
|
time_offset_pre_18/1,
|
||||||
|
time_offset_post_18/1,
|
||||||
convert_time_unit/3,
|
convert_time_unit/3,
|
||||||
|
convert_time_unit_pre_18/3,
|
||||||
|
convert_time_unit_post_18/3,
|
||||||
timestamp/0,
|
timestamp/0,
|
||||||
|
timestamp_pre_18/0,
|
||||||
|
timestamp_post_18/0,
|
||||||
unique_integer/0,
|
unique_integer/0,
|
||||||
|
unique_integer_pre_18/0,
|
||||||
|
unique_integer_post_18/0,
|
||||||
unique_integer/1,
|
unique_integer/1,
|
||||||
|
unique_integer_pre_18/1,
|
||||||
|
unique_integer_post_18/1,
|
||||||
monitor/2,
|
monitor/2,
|
||||||
system_info/1,
|
system_info/1,
|
||||||
system_flag/2]).
|
system_flag/2]).
|
||||||
|
|
||||||
monotonic_time() ->
|
monotonic_time() ->
|
||||||
try
|
code_version:update(?MODULE),
|
||||||
erlang:monotonic_time()
|
time_compat:monotonic_time().
|
||||||
catch
|
|
||||||
error:undef ->
|
monotonic_time_post_18() ->
|
||||||
%% Use Erlang system time as monotonic time
|
erlang:monotonic_time().
|
||||||
erlang_system_time_fallback()
|
|
||||||
end.
|
monotonic_time_pre_18() ->
|
||||||
|
erlang_system_time_fallback().
|
||||||
|
|
||||||
monotonic_time(Unit) ->
|
monotonic_time(Unit) ->
|
||||||
|
code_version:update(?MODULE),
|
||||||
|
time_compat:monotonic_time(Unit).
|
||||||
|
|
||||||
|
monotonic_time_post_18(Unit) ->
|
||||||
try
|
try
|
||||||
erlang:monotonic_time(Unit)
|
erlang:monotonic_time(Unit)
|
||||||
catch
|
catch
|
||||||
error:badarg ->
|
error:badarg ->
|
||||||
erlang:error(badarg, [Unit]);
|
erlang:error(badarg, [Unit])
|
||||||
error:undef ->
|
end.
|
||||||
|
|
||||||
|
monotonic_time_pre_18(Unit) ->
|
||||||
%% Use Erlang system time as monotonic time
|
%% Use Erlang system time as monotonic time
|
||||||
STime = erlang_system_time_fallback(),
|
STime = erlang_system_time_fallback(),
|
||||||
try
|
try
|
||||||
convert_time_unit_fallback(STime, native, Unit)
|
convert_time_unit_fallback(STime, native, Unit)
|
||||||
catch
|
catch
|
||||||
error:bad_time_unit -> erlang:error(badarg, [Unit])
|
error:bad_time_unit -> erlang:error(badarg, [Unit])
|
||||||
end
|
|
||||||
end.
|
end.
|
||||||
|
|
||||||
erlang_system_time() ->
|
erlang_system_time() ->
|
||||||
try
|
code_version:update(?MODULE),
|
||||||
erlang:system_time()
|
time_compat:erlang_system_time().
|
||||||
catch
|
|
||||||
error:undef ->
|
erlang_system_time_post_18() ->
|
||||||
erlang_system_time_fallback()
|
erlang:system_time().
|
||||||
end.
|
|
||||||
|
erlang_system_time_pre_18() ->
|
||||||
|
erlang_system_time_fallback().
|
||||||
|
|
||||||
erlang_system_time(Unit) ->
|
erlang_system_time(Unit) ->
|
||||||
|
code_version:update(?MODULE),
|
||||||
|
time_compat:erlang_system_time(Unit).
|
||||||
|
|
||||||
|
erlang_system_time_post_18(Unit) ->
|
||||||
try
|
try
|
||||||
erlang:system_time(Unit)
|
erlang:system_time(Unit)
|
||||||
catch
|
catch
|
||||||
error:badarg ->
|
error:badarg ->
|
||||||
erlang:error(badarg, [Unit]);
|
erlang:error(badarg, [Unit])
|
||||||
error:undef ->
|
end.
|
||||||
|
|
||||||
|
erlang_system_time_pre_18(Unit) ->
|
||||||
STime = erlang_system_time_fallback(),
|
STime = erlang_system_time_fallback(),
|
||||||
try
|
try
|
||||||
convert_time_unit_fallback(STime, native, Unit)
|
convert_time_unit_fallback(STime, native, Unit)
|
||||||
catch
|
catch
|
||||||
error:bad_time_unit -> erlang:error(badarg, [Unit])
|
error:bad_time_unit -> erlang:error(badarg, [Unit])
|
||||||
end
|
|
||||||
end.
|
end.
|
||||||
|
|
||||||
os_system_time() ->
|
os_system_time() ->
|
||||||
try
|
code_version:update(?MODULE),
|
||||||
os:system_time()
|
time_compat:os_system_time().
|
||||||
catch
|
|
||||||
error:undef ->
|
os_system_time_post_18() ->
|
||||||
os_system_time_fallback()
|
os:system_time().
|
||||||
end.
|
|
||||||
|
os_system_time_pre_18() ->
|
||||||
|
os_system_time_fallback().
|
||||||
|
|
||||||
os_system_time(Unit) ->
|
os_system_time(Unit) ->
|
||||||
|
code_version:update(?MODULE),
|
||||||
|
time_compat:os_system_time(Unit).
|
||||||
|
|
||||||
|
os_system_time_post_18(Unit) ->
|
||||||
try
|
try
|
||||||
os:system_time(Unit)
|
os:system_time(Unit)
|
||||||
catch
|
catch
|
||||||
error:badarg ->
|
error:badarg ->
|
||||||
erlang:error(badarg, [Unit]);
|
erlang:error(badarg, [Unit])
|
||||||
error:undef ->
|
end.
|
||||||
|
|
||||||
|
os_system_time_pre_18(Unit) ->
|
||||||
STime = os_system_time_fallback(),
|
STime = os_system_time_fallback(),
|
||||||
try
|
try
|
||||||
convert_time_unit_fallback(STime, native, Unit)
|
convert_time_unit_fallback(STime, native, Unit)
|
||||||
catch
|
catch
|
||||||
error:bad_time_unit -> erlang:error(badarg, [Unit])
|
error:bad_time_unit -> erlang:error(badarg, [Unit])
|
||||||
end
|
|
||||||
end.
|
end.
|
||||||
|
|
||||||
time_offset() ->
|
time_offset() ->
|
||||||
try
|
code_version:update(?MODULE),
|
||||||
erlang:time_offset()
|
time_compat:time_offset().
|
||||||
catch
|
|
||||||
error:undef ->
|
time_offset_post_18() ->
|
||||||
|
erlang:time_offset().
|
||||||
|
|
||||||
|
time_offset_pre_18() ->
|
||||||
%% Erlang system time and Erlang monotonic
|
%% Erlang system time and Erlang monotonic
|
||||||
%% time are always aligned
|
%% time are always aligned
|
||||||
0
|
0.
|
||||||
end.
|
|
||||||
|
|
||||||
time_offset(Unit) ->
|
time_offset(Unit) ->
|
||||||
|
code_version:update(?MODULE),
|
||||||
|
time_compat:time_offset(Unit).
|
||||||
|
|
||||||
|
time_offset_post_18(Unit) ->
|
||||||
try
|
try
|
||||||
erlang:time_offset(Unit)
|
erlang:time_offset(Unit)
|
||||||
catch
|
catch
|
||||||
error:badarg ->
|
error:badarg ->
|
||||||
erlang:error(badarg, [Unit]);
|
erlang:error(badarg, [Unit])
|
||||||
error:undef ->
|
end.
|
||||||
|
|
||||||
|
time_offset_pre_18(Unit) ->
|
||||||
try
|
try
|
||||||
_ = integer_time_unit(Unit)
|
_ = integer_time_unit(Unit)
|
||||||
catch
|
catch
|
||||||
|
|
@ -154,48 +220,62 @@ time_offset(Unit) ->
|
||||||
end,
|
end,
|
||||||
%% Erlang system time and Erlang monotonic
|
%% Erlang system time and Erlang monotonic
|
||||||
%% time are always aligned
|
%% time are always aligned
|
||||||
0
|
0.
|
||||||
end.
|
|
||||||
|
|
||||||
convert_time_unit(Time, FromUnit, ToUnit) ->
|
convert_time_unit(Time, FromUnit, ToUnit) ->
|
||||||
|
code_version:update(?MODULE),
|
||||||
|
time_compat:convert_time_unit(Time, FromUnit, ToUnit).
|
||||||
|
|
||||||
|
convert_time_unit_post_18(Time, FromUnit, ToUnit) ->
|
||||||
try
|
try
|
||||||
erlang:convert_time_unit(Time, FromUnit, ToUnit)
|
erlang:convert_time_unit(Time, FromUnit, ToUnit)
|
||||||
catch
|
catch
|
||||||
error:undef ->
|
error:Error ->
|
||||||
|
erlang:error(Error, [Time, FromUnit, ToUnit])
|
||||||
|
end.
|
||||||
|
|
||||||
|
convert_time_unit_pre_18(Time, FromUnit, ToUnit) ->
|
||||||
try
|
try
|
||||||
convert_time_unit_fallback(Time, FromUnit, ToUnit)
|
convert_time_unit_fallback(Time, FromUnit, ToUnit)
|
||||||
catch
|
catch
|
||||||
_:_ ->
|
_:_ ->
|
||||||
erlang:error(badarg, [Time, FromUnit, ToUnit])
|
erlang:error(badarg, [Time, FromUnit, ToUnit])
|
||||||
end;
|
|
||||||
error:Error ->
|
|
||||||
erlang:error(Error, [Time, FromUnit, ToUnit])
|
|
||||||
end.
|
end.
|
||||||
|
|
||||||
timestamp() ->
|
timestamp() ->
|
||||||
try
|
code_version:update(?MODULE),
|
||||||
erlang:timestamp()
|
time_compat:timestamp().
|
||||||
catch
|
|
||||||
error:undef ->
|
timestamp_post_18() ->
|
||||||
erlang:now()
|
erlang:timestamp().
|
||||||
end.
|
|
||||||
|
timestamp_pre_18() ->
|
||||||
|
erlang:now().
|
||||||
|
|
||||||
unique_integer() ->
|
unique_integer() ->
|
||||||
try
|
code_version:update(?MODULE),
|
||||||
erlang:unique_integer()
|
time_compat:unique_integer().
|
||||||
catch
|
|
||||||
error:undef ->
|
unique_integer_post_18() ->
|
||||||
|
erlang:unique_integer().
|
||||||
|
|
||||||
|
unique_integer_pre_18() ->
|
||||||
{MS, S, US} = erlang:now(),
|
{MS, S, US} = erlang:now(),
|
||||||
(MS*1000000+S)*1000000+US
|
(MS*1000000+S)*1000000+US.
|
||||||
end.
|
|
||||||
|
|
||||||
unique_integer(Modifiers) ->
|
unique_integer(Modifiers) ->
|
||||||
|
code_version:update(?MODULE),
|
||||||
|
time_compat:unique_integer(Modifiers).
|
||||||
|
|
||||||
|
unique_integer_post_18(Modifiers) ->
|
||||||
try
|
try
|
||||||
erlang:unique_integer(Modifiers)
|
erlang:unique_integer(Modifiers)
|
||||||
catch
|
catch
|
||||||
error:badarg ->
|
error:badarg ->
|
||||||
erlang:error(badarg, [Modifiers]);
|
erlang:error(badarg, [Modifiers])
|
||||||
error:undef ->
|
end.
|
||||||
|
|
||||||
|
unique_integer_pre_18(Modifiers) ->
|
||||||
case is_valid_modifier_list(Modifiers) of
|
case is_valid_modifier_list(Modifiers) of
|
||||||
true ->
|
true ->
|
||||||
%% now() converted to an integer
|
%% now() converted to an integer
|
||||||
|
|
@ -206,7 +286,6 @@ unique_integer(Modifiers) ->
|
||||||
(MS*1000000+S)*1000000+US;
|
(MS*1000000+S)*1000000+US;
|
||||||
false ->
|
false ->
|
||||||
erlang:error(badarg, [Modifiers])
|
erlang:error(badarg, [Modifiers])
|
||||||
end
|
|
||||||
end.
|
end.
|
||||||
|
|
||||||
monitor(Type, Item) ->
|
monitor(Type, Item) ->
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue