You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 

1297 lines
49 KiB

-module(eSync).
-behaviour(gen_server).
-compile(inline).
-compile({inline_size, 128}).
-define(IIF(Cond, Ret1, Ret2), (case Cond of true -> Ret1; _ -> Ret2 end)).
-define(LOG_ON(Val), Val == true; Val == all; Val == skip_success; is_list(Val), Val =/= []).
-define(Log, log).
-define(compileCmd, compileCmd).
-define(extraDirs, extraDirs).
-define(descendant, descendant).
-define(onMSyncFun, onMSyncFun).
-define(onCSyncFun, onCSyncFun).
-define(swSyncNode, swSyncNode).
-define(isJustMem, isJustMem).
-define(debugInfoKeyFun, debugInfoKeyFun).
-define(DefCfgList, [{?Log, all}, {?compileCmd, undefined}, {?extraDirs, undefined}, {?descendant, fix}, {?onMSyncFun, undefined}, {?onCSyncFun, undefined}, {?swSyncNode, false}, {?isJustMem, false}, {?debugInfoKeyFun, undefined}]).
-define(esCfgSync, esCfgSync).
-define(rootSrcDir, <<"src">>).
-export([
start/2,
start/0,
stop/0,
run/0
]).
%% API
-export([
start_link/0,
rescan/0,
pause/0,
unpause/0,
setLog/1,
getLog/0,
curInfo/0,
getOnMSync/0,
setOnMSync/1,
getOnCSync/0,
setOnCSync/1,
swSyncNode/1
]).
%% gen_server callbacks
-export([
init/1,
handle_call/3,
handle_cast/2,
handle_info/2,
terminate/2
]).
start(_StartType, _StartArgs) ->
start_link().
start() ->
application:ensure_all_started(eSync).
stop() ->
application:stop(eSync).
run() ->
case start() of
{ok, _Started} ->
unpause(),
ok;
{error, Reason} ->
logErrors("start eSync error:~p~n", [Reason])
end.
-define(SERVER, ?MODULE).
-define(None, 0).
-record(state, {
status = wait
, port = undefined
, onMSyncFun = undefined
, onCSyncFun = undefined
, swSyncNode = false
, srcFiles = #{} :: map()
, hrlFiles = #{} :: map()
, configs = #{} :: map()
, beams = #{} :: map()
}).
%% ************************************ API start ***************************
rescan() ->
es_gen_ipc:cast(?SERVER, miRescan),
logSuccess("start rescaning source files..."),
ok.
unpause() ->
es_gen_ipc:cast(?SERVER, miUnpause),
ok.
pause() ->
es_gen_ipc:cast(?SERVER, miPause),
logSuccess("Pausing eSync. Call eSync:run() to restart"),
ok.
curInfo() ->
es_gen_ipc:call(?SERVER, miCurInfo).
setLog(T) when ?LOG_ON(T) ->
setEnv(log, T),
loadCfg(),
logSuccess("Console Notifications Enabled"),
ok;
setLog(_) ->
setEnv(log, none),
loadCfg(),
logSuccess("Console Notifications Disabled"),
ok.
getLog() ->
?esCfgSync:getv(log).
swSyncNode(IsSync) ->
es_gen_ipc:cast(?SERVER, {miSyncNode, IsSync}),
ok.
getOnMSync() ->
es_gen_ipc:call(?SERVER, miGetOnMSync).
setOnMSync(Fun) ->
es_gen_ipc:call(?SERVER, {miSetOnMSync, Fun}).
getOnCSync() ->
es_gen_ipc:call(?SERVER, miGetOnCSync).
setOnCSync(Fun) ->
es_gen_ipc:call(?SERVER, {miSetOnCSync, Fun}).
%% ************************************ API end ***************************
start_link() ->
gen_server:start_link({local, ?SERVER}, ?MODULE, [], []).
init([]) ->
erlang:process_flag(trap_exit, true),
loadCfg(),
erlang:send_after(0, self(), doAfter),
{ok, #state{onMSyncFun = ?esCfgSync:getv(?onMSyncFun), onCSyncFun = ?esCfgSync:getv(?onCSyncFun), swSyncNode = ?esCfgSync:getv(?swSyncNode)}}.
handle_call(miGetOnMSync, _, #state{onMSyncFun = OnMSyncFun} = State) ->
{reply, OnMSyncFun, State};
handle_call({miSetOnMSync, Fun}, _, State) ->
{reply, ok, State#state{onMSyncFun = Fun}};
handle_call(miGetOnCSync, _, #state{onCSyncFun = OnCSyncFun} = State) ->
{reply, OnCSyncFun, State};
handle_call({miSetOnCSync, Fun}, _, State) ->
{reply, ok, State#state{onCSyncFun = Fun}};
handle_call(miCurInfo, _, State) ->
{reply, {erlang:get(), State}, State};
handle_call(_Request, _, State) ->
{reply, ok, State}.
handle_cast(miPause, State) ->
{noreply, State#state{status = pause}};
handle_cast(miUnpause, State) ->
{noreply, State#state{status = running}};
handle_cast({miSyncNode, IsSync}, State) ->
case IsSync of
true ->
{noreply, State#state{swSyncNode = true}};
_ ->
{noreply, State#state{swSyncNode = false}}
end;
handle_cast(miRescan, State) ->
{Srcs, Hrls, Configs, Beams} = collSrcFiles(false),
{noreply, State#state{srcFiles = Srcs, hrlFiles = Hrls, configs = Configs, beams = Beams}, hibernate};
handle_cast(_Msg, State) ->
{noreply, State}.
handle_info({Port, {data, Data}}, #state{status = Status, srcFiles = Srcs, hrlFiles = Hrls, configs = Configs, beams = Beams, onMSyncFun = OnMSyncFun, onCSyncFun = OnCSyncFun, swSyncNode = SwSyncNode} = State) ->
case Status of
running ->
FileList = binary:split(Data, <<"\r\n">>, [global]),
%% 收集改动了beam hrl src 文件 然后执行相应的逻辑
{CBeams, CConfigs, CHrls, CSrcs, NewSrcs, NewHrls, NewConfigs, NewBeams} = classifyChangeFile(FileList, [], [], #{}, #{}, Srcs, Hrls, Configs, Beams),
fireOnSync(OnCSyncFun, CConfigs),
reloadChangedMod(CBeams, SwSyncNode, OnMSyncFun, []),
case ?esCfgSync:getv(?compileCmd) of
undefined ->
LastCHrls = collIncludeCHrls(maps:keys(CHrls), NewHrls, CHrls, #{}),
NReSrcs = collIncludeCErls(maps:keys(LastCHrls), NewSrcs, CSrcs, #{}),
recompileChangeSrcFile(maps:iterator(NReSrcs), SwSyncNode),
{noreply, State#state{srcFiles = NewSrcs, hrlFiles = NewHrls, configs = NewConfigs, beams = NewBeams}};
CmdStr ->
case maps:size(CSrcs) > 0 orelse CHrls =/= [] of
true ->
RetStr = os:cmd(CmdStr),
RetList = string:split(RetStr, "\n", all),
logSuccess("compile cmd:~p ~n", [CmdStr]),
logSuccess("the result: ~n ", []),
[
begin
logSuccess("~p ~n", [OneRet])
end || OneRet <- RetList, OneRet =/= []
],
ok;
_ ->
ignore
end,
{noreply, State}
end;
pause ->
{noreply, State};
wait ->
case Data of
<<"init">> ->
%% port启动成功 先发送监听目录配置
{AddExtraSrcDirs, AddOnlySrcDirs, OnlySrcDirs, DelSrcDirs} = mergeExtraDirs(false),
AddExtraStr = string:join([filename:nativename(OneDir) || OneDir <- AddExtraSrcDirs], "|"),
AddOnlyStr = string:join([filename:nativename(OneDir) || OneDir <- AddOnlySrcDirs], "|"),
OnlyStr = string:join([filename:nativename(OneDir) || OneDir <- OnlySrcDirs], "|"),
DelStr = string:join([filename:nativename(OneDir) || OneDir <- DelSrcDirs], "|"),
AllStr = string:join([AddExtraStr, AddOnlyStr, OnlyStr, DelStr], "\r\n"),
erlang:port_command(Port, AllStr),
logSuccess("eSync connect fileSync success..."),
%% 然后收集一下监听目录下的src文件
{BSrcs, BHrls, BConfigs, BBeams} = collSrcFiles(true),
{noreply, State#state{status = running, srcFiles = BSrcs, hrlFiles = BHrls, configs = BConfigs, beams = BBeams}, hibernate};
_ ->
logErrors("error, eSync receive unexpect port msg ~p~n", [Data]),
{noreply, State}
end
end;
handle_info({Port, closed}, #state{port = Port} = _State) ->
logErrors("eSync receive port closed ~n"),
{stop, port_close, _State};
handle_info({'EXIT', Port, Reason}, #state{port = Port} = _State) ->
logErrors("eSync receive port exit Reason:~p ~n", [Reason]),
{stop, {port_EXIT, Reason}, _State};
handle_info({Port, {exit_status, Status}}, #state{port = Port} = _State) ->
logErrors("eSync receive port exit_status Status:~p ~p ~n", [Status, Port]),
{stop, {port_exit_status, Status}, _State};
handle_info({'EXIT', _Pid, _Reason}, _State) ->
{noreply, _State};
handle_info(doAfter, State) ->
%% 启动port 发送监听目录信息
PortName = fileSyncPath("fileSync"),
Opts = [{packet, 4}, binary, exit_status, use_stdio],
Port = erlang:open_port({spawn_executable, PortName}, Opts),
{noreply, State#state{port = Port}, 4000};
handle_info(timeout, State) ->
logErrors("failed to connect the fileSync to stop state:~p ~n", [State]),
{stop, waitConnOver, State};
handle_info(_Msg, State) ->
{noreply, State}.
terminate(_Reason, _State) ->
ok.
%% ************************************************* utils start *******************************************************
getModSrcDir(Module) ->
case code:is_loaded(Module) of
{file, _} ->
try
%% Get some module info...
Props = Module:module_info(compile),
Source = proplists:get_value(source, Props, ""),
%% Ensure that the file exists, is a decendent of the tree, and how to deal with that
IsFile = filelib:is_regular(Source),
IsDescendant = isDescendent(Source),
Descendant = ?esCfgSync:getv(?descendant),
LastSource =
case {IsFile, IsDescendant, Descendant} of
%% is file and descendant, we're good to go
{true, true, _} -> Source;
%% is not a descendant, but we allow them, so good to go
{true, false, allow} -> Source;
%% is not a descendant, and we fix non-descendants, so let's fix it
{_, false, fix} -> fixDescendantSource(Source, IsFile);
%% Anything else, and we don't know what to do, so let's just bail.
_ -> undefined
end,
case LastSource of
undefined ->
undefined;
_ ->
%% Get the source dir...
Dir = filename:dirname(LastSource),
getSrcDir(Dir)
end
catch _ : _ ->
undefined
end;
_ ->
undefined
end.
getModOpts(Module) ->
case code:is_loaded(Module) of
{file, _} ->
try
Props = Module:module_info(compile),
BeamDir = filename:dirname(code:which(Module)),
Options1 = proplists:get_value(options, Props, []),
%% transform `outdir'
Options2 = transformOutdir(BeamDir, Options1),
Options3 = ensureInclude(Options2),
%% transform the include directories
Options4 = transformAllIncludes(Module, BeamDir, Options3),
%% maybe_add_compile_info
Options5 = maybeAddCompileInfo(Options4),
%% add filetype to options (DTL, LFE, erl, etc)
Options6 = addFileType(Module, Options5),
Options7 = lists:keyreplace(debug_info_key, 1, Options6, debugInfoKeyFun()),
{ok, Options7}
catch ExType:Error ->
logWarnings("~p:0: ~p looking for options: ~p. ~n", [Module, ExType, Error]),
undefined
end;
_ ->
undefined
end.
tryGetModOpts(Module) ->
try
Props = Module:module_info(compile),
BeamDir = filename:dirname(code:which(Module)),
Options1 = proplists:get_value(options, Props, []),
%% transform `outdir'
Options2 = transformOutdir(BeamDir, Options1),
Options3 = ensureInclude(Options2),
%% transform the include directories
Options4 = transformAllIncludes(Module, BeamDir, Options3),
%% maybe_add_compile_info
Options5 = maybeAddCompileInfo(Options4),
%% add filetype to options (DTL, LFE, erl, etc)
Options6 = addFileType(Module, Options5),
Options7 = lists:keyreplace(debug_info_key, 1, Options6, debugInfoKeyFun()),
{ok, Options7}
catch _ExType:_Error ->
undefiend
end.
tryGetSrcOpts(SrcDir) ->
%% Then we dig back through the parent directories until we find our include directory
NewDirName = filename:dirname(SrcDir),
case getOptions(NewDirName) of
{ok, _Options} = Opts ->
Opts;
_ ->
BaseName = filename:basename(SrcDir),
IsBaseSrcDir = BaseName == ?rootSrcDir,
case NewDirName =/= SrcDir andalso not IsBaseSrcDir of
true ->
tryGetSrcOpts(NewDirName);
_ when IsBaseSrcDir ->
try filelib:fold_files(SrcDir, ".*\\.(erl|dtl|lfe|ex)$", true,
fun(OneFile, Acc) ->
Mod = binary_to_atom(filename:basename(OneFile, filename:extension(OneFile))),
case tryGetModOpts(Mod) of
{ok, _Options} = Opts ->
throw(Opts);
_ ->
Acc
end
end, undefined)
catch
{ok, _Options} = Opts ->
Opts;
_ExType:_Error ->
logWarnings("looking src options error ~p:~p. ~n", [_ExType, _Error]),
undefined
end;
_ ->
undefined
end
end.
transformOutdir(BeamDir, Options) ->
[{outdir, BeamDir} | proplists:delete(outdir, Options)].
ensureInclude(Options) ->
case proplists:get_value(i, Options) of
undefined -> [{i, "include"} | Options];
_ -> Options
end.
transformAllIncludes(Module, BeamDir, Options) ->
[begin
case Opt of
{i, IncludeDir} ->
{ok, SrcDir} = getModSrcDir(Module),
{ok, IncludeDir2} = determineIncludeDir(IncludeDir, BeamDir, SrcDir),
{i, IncludeDir2};
_ ->
Opt
end
end || Opt <- Options].
maybeAddCompileInfo(Options) ->
case lists:member(compile_info, Options) of
true -> Options;
false -> addCompileInfo(Options)
end.
addCompileInfo(Options) ->
CompInfo = [{K, V} || {K, V} <- Options, lists:member(K, [outdir, i])],
[{compile_info, CompInfo} | Options].
addFileType(Module, Options) ->
Type = getFileType(Module),
[{type, Type} | Options].
%% This will check if the given module or source file is an ErlyDTL template.
%% Currently, this is done by checking if its reported source path ends with
%% ".dtl.erl".
getFileType(Module) when is_atom(Module) ->
Props = Module:module_info(compile),
Source = proplists:get_value(source, Props, ""),
getFileType(Source);
getFileType(Source) ->
Ext = filename:extension(Source),
Root = filename:rootname(Source),
SecondExt = filename:extension(Root),
case Ext of
<<".erl">> when SecondExt =:= <<".dtl">> -> dtl;
<<".dtl">> -> dtl;
<<".erl">> -> erl;
<<".lfe">> -> lfe;
<<".ex">> -> elixir;
".erl" when SecondExt =:= ".dtl" -> dtl;
".dtl" -> dtl;
".erl" -> erl;
".lfe" -> lfe;
".ex" -> elixir
end.
%% This will search back to find an appropriate include directory, by
%% searching further back than "..". Instead, it will extract the basename
%% (probably "include" from the include pathfile, and then search backwards in
%% the directory tree until it finds a directory with the same basename found
%% above.
determineIncludeDir(IncludeDir, BeamDir, SrcDir) ->
IncludeBase = filename:basename(IncludeDir),
case determineIncludeDirFromBeamDir(IncludeBase, IncludeDir, BeamDir) of
{ok, _Dir} = RetD -> RetD;
undefined ->
{ok, Cwd} = file:get_cwd(),
% Cwd2 = normalizeCaseWindowsDir(Cwd),
% SrcDir2 = normalizeCaseWindowsDir(SrcDir),
% IncludeBase2 = normalizeCaseWindowsDir(IncludeBase),
case findIncludeDirFromAncestors(SrcDir, Cwd, IncludeBase) of
{ok, _Dir} = RetD -> RetD;
undefined -> {ok, IncludeDir} %% Failed, just stick with original
end
end.
%% First try to see if we have an include file alongside our ebin directory, which is typically the case
determineIncludeDirFromBeamDir(IncludeBase, IncludeDir, BeamDir) ->
BeamBasedIncDir = filename:join(filename:dirname(BeamDir), IncludeBase),
case filelib:is_dir(BeamBasedIncDir) of
true -> {ok, BeamBasedIncDir};
false ->
BeamBasedIncDir2 = filename:join(filename:dirname(BeamDir), IncludeDir),
case filelib:is_dir(BeamBasedIncDir2) of
true -> {ok, BeamBasedIncDir2};
_ ->
undefined
end
end.
%% get the src dir
getRootSrcDirFromSrcDir(SrcDir) ->
NewDirName = filename:dirname(SrcDir),
BaseName = filename:basename(SrcDir),
case BaseName of
?rootSrcDir ->
NewDirName;
_ ->
case NewDirName =/= SrcDir of
true ->
getRootSrcDirFromSrcDir(NewDirName);
_ ->
undefined
end
end.
%% Then we dig back through the parent directories until we find our include directory
findIncludeDirFromAncestors(Cwd, Cwd, _) -> undefined;
findIncludeDirFromAncestors("/", _, _) -> undefined;
findIncludeDirFromAncestors(".", _, _) -> undefined;
findIncludeDirFromAncestors("", _, _) -> undefined;
findIncludeDirFromAncestors(Dir, Cwd, IncludeBase) ->
NewDirName = filename:dirname(Dir),
AttemptDir = filename:join(NewDirName, IncludeBase),
case filelib:is_dir(AttemptDir) of
true ->
{ok, AttemptDir};
false ->
case NewDirName =/= Dir of
true ->
findIncludeDirFromAncestors(NewDirName, Cwd, IncludeBase);
_ ->
undefined
end
end.
% normalizeCaseWindowsDir(Dir) ->
% case os:type() of
% {win32, _} -> Dir; %string:to_lower(Dir);
% {unix, _} -> Dir
% end.
%% This is an attempt to intelligently fix paths in modules when a
%% release is moved. Essentially, it takes a module name and its original path
%% from Module:module_info(compile), say
%% "/some/original/path/site/src/pages/somepage.erl", and then breaks down the
%% path one by one prefixing it with the current working directory until it
%% either finds a match, or fails. If it succeeds, it returns the Path to the
%% new Source file.
fixDescendantSource([], _IsFile) ->
undefined;
fixDescendantSource(Path, IsFile) ->
{ok, Cwd} = file:get_cwd(),
PathParts = filename:split(Path),
case makeDescendantSource(PathParts, Cwd) of
undefined -> case IsFile of true -> Path; _ -> undefined end;
FoundPath -> FoundPath
end.
makeDescendantSource([], _Cwd) ->
undefined;
makeDescendantSource([_ | T], Cwd) ->
PathAttempt = filename:join([Cwd | T]),
case filelib:is_regular(PathAttempt) of
true -> PathAttempt;
false -> makeDescendantSource(T, Cwd)
end.
isDescendent(Path) ->
{ok, Cwd} = file:get_cwd(),
lists:sublist(Path, length(Cwd)) == Cwd.
%% @private Find the src directory for the specified Directory; max 15 iterations
getSrcDir(Dir) ->
getSrcDir(Dir, 15).
getSrcDir(_Dir, 0) ->
undefined;
getSrcDir(Dir, Ctr) ->
HasCode = filelib:wildcard("*.erl", Dir) /= [] orelse
filelib:wildcard("*.hrl", Dir) /= [] orelse
filelib:wildcard("*.dtl", Dir) /= [] orelse
filelib:wildcard("*.lfe", Dir) /= [] orelse
filelib:wildcard("*.ex", Dir) /= [],
if
HasCode -> {ok, Dir};
true -> getSrcDir(filename:dirname(Dir), Ctr - 1)
end.
mergeExtraDirs(IsAddPath) ->
case ?esCfgSync:getv(?extraDirs) of
undefined ->
{[], [], [], []};
ExtraList ->
FunMerge =
fun(OneExtra, {AddExtraDirs, AddOnlyDirs, OnlyDirs, DelDirs} = AllAcc) ->
case OneExtra of
{addExtra, DirsAndOpts} ->
Adds =
[
begin
case IsAddPath of
true ->
case proplists:get_value(outdir, Opts) of
undefined ->
true;
Path ->
ok = filelib:ensure_dir(Path),
true = code:add_pathz(Path)
end;
_ ->
ignore
end,
filename:absname(Dir)
end || {Dir, Opts} <- DirsAndOpts
],
setelement(1, AllAcc, Adds ++ AddExtraDirs);
{addOnly, DirsAndOpts} ->
Adds =
[
begin
case IsAddPath of
true ->
case proplists:get_value(outdir, Opts) of
undefined ->
true;
Path ->
ok = filelib:ensure_dir(Path),
true = code:add_pathz(Path)
end;
_ ->
ignore
end,
filename:absname(Dir)
end || {Dir, Opts} <- DirsAndOpts
],
setelement(2, AllAcc, Adds ++ AddOnlyDirs);
{only, DirsAndOpts} ->
Onlys =
[
begin
case IsAddPath of
true ->
case proplists:get_value(outdir, Opts) of
undefined ->
true;
Path ->
ok = filelib:ensure_dir(Path),
true = code:add_pathz(Path)
end;
_ ->
ignore
end,
filename:absname(Dir)
end || {Dir, Opts} <- DirsAndOpts
],
setelement(3, AllAcc, Onlys ++ OnlyDirs);
{del, DirsAndOpts} ->
Dels =
[
begin
filename:absname(Dir)
end || {Dir, _Opts} <- DirsAndOpts
],
setelement(4, AllAcc, Dels ++ DelDirs)
end
end,
lists:foldl(FunMerge, {[], [], [], []}, ExtraList)
end.
-define(RegExp, <<".*\\.(erl|hrl|beam|config|dtl|lfe|ex)$">>).
collSrcFiles(IsAddPath) ->
{AddExtraSrcDirs, AddOnlySrcDirs, OnlySrcDirs, DelSrcDirs} = mergeExtraDirs(IsAddPath),
CollFiles = filelib:fold_files(filename:absname(<<"./">>), ?RegExp, true,
fun(OneFile, {Srcs, Hrls, Configs, Beams} = Acc) ->
case isOnlyDir(OnlySrcDirs, OneFile) andalso (not isDelDir(DelSrcDirs, OneFile)) of
true ->
MTimeSec = dateTimeToSec(filelib:last_modified(OneFile)),
case filename:extension(OneFile) of
<<".beam">> ->
BeamMod = binary_to_atom(filename:basename(OneFile, <<".beam">>)),
setelement(4, Acc, Beams#{BeamMod => MTimeSec});
<<".config">> ->
setelement(3, Acc, Configs#{OneFile => MTimeSec});
<<".hrl">> ->
setelement(2, Acc, Hrls#{OneFile => MTimeSec});
<<>> ->
Acc;
_ ->
RootSrcDir =
case getRootSrcDirFromSrcDir(OneFile) of
undefined ->
filename:dirname(OneFile);
RetSrcDir ->
RetSrcDir
end,
case getOptions(RootSrcDir) of
undefined ->
Mod = binary_to_atom(filename:basename(OneFile, filename:extension(OneFile))),
case getModOpts(Mod) of
{ok, Options} ->
setOptions(RootSrcDir, Options);
_ ->
ignore
end;
_ ->
ignore
end,
setelement(1, Acc, Srcs#{OneFile => MTimeSec})
end;
_ ->
Acc
end
end, {#{}, #{}, #{}, #{}}),
FunCollAddExtra =
fun(OneDir, FilesAcc) ->
filelib:fold_files(?IIF(is_list(OneDir), list_to_binary(OneDir), OneDir), ?RegExp, true,
fun(OneFile, {Srcs, Hrls, Configs, Beams} = Acc) ->
MTimeSec = dateTimeToSec(filelib:last_modified(OneFile)),
case filename:extension(OneFile) of
<<".beam">> ->
BeamMod = binary_to_atom(filename:basename(OneFile, <<".beam">>)),
setelement(4, Acc, Beams#{BeamMod => MTimeSec});
<<".config">> ->
setelement(3, Acc, Configs#{OneFile => MTimeSec});
<<".hrl">> ->
setelement(2, Acc, Hrls#{OneFile => MTimeSec});
<<>> ->
Acc;
_ ->
setelement(1, Acc, Srcs#{OneFile => MTimeSec})
end
end, FilesAcc)
end,
AddExtraCollFiles = lists:foldl(FunCollAddExtra, CollFiles, AddExtraSrcDirs),
FunCollAddOnly =
fun(OneDir, FilesAcc) ->
filelib:fold_files(?IIF(is_list(OneDir), list_to_binary(OneDir), OneDir), ?RegExp, false,
fun(OneFile, {Srcs, Hrls, Configs, Beams} = Acc) ->
MTimeSec = dateTimeToSec(filelib:last_modified(OneFile)),
case filename:extension(OneFile) of
<<".beam">> ->
BeamMod = binary_to_atom(filename:basename(OneFile, <<".beam">>)),
setelement(4, Acc, Beams#{BeamMod => MTimeSec});
<<".config">> ->
setelement(3, Acc, Configs#{OneFile => MTimeSec});
<<".hrl">> ->
setelement(2, Acc, Hrls#{OneFile => MTimeSec});
<<>> ->
Acc;
_ ->
setelement(1, Acc, Srcs#{OneFile => MTimeSec})
end
end, FilesAcc)
end,
lists:foldl(FunCollAddOnly, AddExtraCollFiles, AddOnlySrcDirs).
isOnlyDir([], _) ->
true;
isOnlyDir(ReplaceDirs, SrcDir) ->
isMatchDir(ReplaceDirs, SrcDir).
isDelDir([], _) ->
false;
isDelDir(ReplaceDirs, SrcDir) ->
isMatchDir(ReplaceDirs, SrcDir).
isMatchDir([], _SrcDir) ->
false;
isMatchDir([SrcDir | _ReplaceDirs], SrcDir) ->
true;
isMatchDir([OneDir | ReplaceDirs], SrcDir) ->
case re:run(SrcDir, OneDir) of
nomatch -> isMatchDir(ReplaceDirs, SrcDir);
_ -> true
end.
getEnv(Var, Default) ->
case application:get_env(eSync, Var) of
{ok, Value} ->
Value;
_ ->
Default
end.
setEnv(Var, Val) ->
ok = application:set_env(eSync, Var, Val).
logSuccess(Format) ->
canLog(success) andalso error_logger:info_msg(Format).
logSuccess(Format, Args) ->
canLog(success) andalso error_logger:info_msg(Format, Args).
logErrors(Format) ->
canLog(errors) andalso error_logger:error_msg(Format).
logErrors(Format, Args) ->
canLog(errors) andalso error_logger:error_msg(Format, Args).
logWarnings(Format, Args) ->
canLog(warnings) andalso error_logger:warning_msg(Format, Args).
canLog(MsgType) ->
case eSync:getLog() of
true -> true;
all -> true;
none -> false;
false -> false;
skip_success -> MsgType == errors orelse MsgType == warnings;
L when is_list(L) -> lists:member(MsgType, L);
_ -> false
end.
%% 注意 map类型的数据不能当做key
-type key() :: atom() | binary() | bitstring() | float() | integer() | list() | tuple().
-type value() :: atom() | binary() | bitstring() | float() | integer() | list() | tuple() | map().
-spec load(term(), [{key(), value()}]) -> ok.
load(Module, KVs) ->
Forms = forms(Module, KVs),
{ok, Module, Bin} = compile:forms(Forms),
code:soft_purge(Module),
{module, Module} = code:load_binary(Module, atom_to_list(Module), Bin),
ok.
forms(Module, KVs) ->
%% -module(Module).
Mod = erl_syntax:attribute(erl_syntax:atom(module), [erl_syntax:atom(Module)]),
%% -export([getv/0]).
ExportList = [erl_syntax:arity_qualifier(erl_syntax:atom(getv), erl_syntax:integer(1))],
Export = erl_syntax:attribute(erl_syntax:atom(export), [erl_syntax:list(ExportList)]),
%% getv(K) -> V
Function = erl_syntax:function(erl_syntax:atom(getv), lookup_clauses(KVs, [])),
[erl_syntax:revert(X) || X <- [Mod, Export, Function]].
lookup_clause(Key, Value) ->
Var = erl_syntax:abstract(Key),
Body = erl_syntax:abstract(Value),
erl_syntax:clause([Var], [], [Body]).
lookup_clause_anon() ->
Var = erl_syntax:variable("_"),
Body = erl_syntax:atom(undefined),
erl_syntax:clause([Var], [], [Body]).
lookup_clauses([], Acc) ->
lists:reverse(lists:flatten([lookup_clause_anon() | Acc]));
lookup_clauses([{Key, Value} | T], Acc) ->
lookup_clauses(T, [lookup_clause(Key, Value) | Acc]).
getOptions(SrcDir) ->
case erlang:get(SrcDir) of
undefined ->
undefined;
Options ->
{ok, Options}
end.
setOptions(SrcDir, Options) ->
case erlang:get(SrcDir) of
undefined ->
erlang:put(SrcDir, Options);
OldOptions ->
NewOptions =
case lists:keytake(compile_info, 1, Options) of
{value, {compile_info, ValList1}, Options1} ->
case lists:keytake(compile_info, 1, OldOptions) of
{value, {compile_info, ValList2}, Options2} ->
[{compile_info, lists:usort(ValList1 ++ ValList2)} | lists:usort(Options1 ++ Options2)];
_ ->
lists:usort(Options ++ OldOptions)
end;
_ ->
lists:usort(Options ++ OldOptions)
end,
erlang:put(SrcDir, NewOptions)
end.
loadCfg() ->
KVs = [{Key, getEnv(Key, DefVal)} || {Key, DefVal} <- ?DefCfgList],
load(?esCfgSync, KVs).
%% ******************************* 加载与编译相关 **********************************************************************
printResults(_Module, SrcFile, [], []) ->
logSuccess("~s Recompiled", [SrcFile]);
printResults(_Module, SrcFile, [], Warnings) ->
formatErrors(fun logWarnings/2, SrcFile, [], Warnings);
printResults(_Module, SrcFile, Errors, Warnings) ->
formatErrors(fun logErrors/2, SrcFile, Errors, Warnings).
%% @private Print error messages in a pretty and user readable way.
formatErrors(LogFun, File, Errors, Warnings) ->
AllErrors1 = lists:sort(lists:flatten([X || {_, X} <- Errors])),
AllErrors2 = [{Line, "Error", Module, Description} || {Line, Module, Description} <- AllErrors1],
AllWarnings1 = lists:sort(lists:flatten([X || {_, X} <- Warnings])),
AllWarnings2 = [{Line, "Warning", Module, Description} || {Line, Module, Description} <- AllWarnings1],
Everything = lists:sort(AllErrors2 ++ AllWarnings2),
FPck =
fun({Line, Prefix, Module, ErrorDescription}) ->
Msg = formatError(Module, ErrorDescription),
LogFun("~s: ~p: ~s: ~s", [File, Line, Prefix, Msg])
end,
[FPck(X) || X <- Everything],
ok.
formatError(Module, ErrorDescription) ->
case erlang:function_exported(Module, format_error, 1) of
true -> Module:format_error(ErrorDescription);
false -> io_lib:format("~s", [ErrorDescription])
end.
fireOnSync(OnSyncFun, Modules) ->
case OnSyncFun of
undefined -> ok;
Funs when is_list(Funs) -> onSyncApplyList(Funs, Modules);
Fun -> onSyncApply(Fun, Modules)
end.
onSyncApplyList(Funs, Modules) ->
[onSyncApply(Fun, Modules) || Fun <- Funs].
onSyncApply({M, F}, Modules) ->
try erlang:apply(M, F, [Modules])
catch
C:R:S ->
logErrors("apply sync fun ~p:~p(~p) error ~p", [M, F, Modules, {C, R, S}])
end;
onSyncApply(Fun, Modules) when is_function(Fun) ->
try Fun(Modules)
catch
C:R:S ->
logErrors("apply sync fun ~p(~p) error ~p", [Fun, Modules, {C, R, S}])
end.
reloadChangedMod([], _SwSyncNode, OnSyncFun, Acc) ->
fireOnSync(OnSyncFun, Acc);
reloadChangedMod([Module | LeftMod], SwSyncNode, OnSyncFun, Acc) ->
case Module == es_gen_ipc orelse code:get_object_code(Module) of
true ->
ignore;
error ->
logErrors("Error loading object code for ~p", [Module]),
reloadChangedMod(LeftMod, SwSyncNode, OnSyncFun, Acc);
{Module, Binary, Filename} ->
case code:load_binary(Module, Filename, Binary) of
{module, Module} ->
logSuccess("Reloaded(Beam changed) Mod: ~s Success", [Module]),
syncLoadModOnAllNodes(SwSyncNode, Module, Binary, changed);
{error, What} ->
logErrors("Reloaded(Beam changed) Mod: ~s Errors Reason:~p", [Module, What])
end,
reloadChangedMod(LeftMod, SwSyncNode, OnSyncFun, [Module | Acc])
end.
getNodes() ->
lists:usort(lists:flatten(nodes() ++ [erpc:call(X, erlang, nodes, []) || X <- nodes()])) -- [node()].
syncLoadModOnAllNodes(false, _Module, _Binary, _Reason) ->
ignore;
syncLoadModOnAllNodes(true, Module, Binary, Reason) ->
%% Get a list of nodes known by this node, plus all attached nodes.
Nodes = getNodes(),
NumNodes = length(Nodes),
[
begin
logSuccess("Do Reloading '~s' on ~p", [Module, Node]),
erpc:call(Node, code, ensure_loaded, [Module]),
case erpc:call(Node, code, which, [Module]) of
Filename when is_binary(Filename) orelse is_list(Filename) ->
%% File exists, overwrite and load into VM.
ok = erpc:call(Node, file, write_file, [Filename, Binary]),
erpc:call(Node, code, purge, [Module]),
case erpc:call(Node, code, load_file, [Module]) of
{module, Module} ->
logSuccess("Reloaded(Beam ~p) Mod:~s and write Success on node:~p", [Reason, Module, Node]);
{error, What} ->
logErrors("Reloaded(Beam ~p) Mod:~s and write Errors on node:~p Reason:~p", [Module, Node, What])
end;
_ ->
%% File doesn't exist, just load into VM.
case erpc:call(Node, code, load_binary, [Module, undefined, Binary]) of
{module, Module} ->
logSuccess("Reloaded(Beam ~p) Mod:~s Success on node:~p", [Reason, Module, Node]);
{error, What} ->
logErrors("Reloaded(Beam ~p) Mod:~s Errors on node:~p Reason:~p", [Reason, Module, Node, What])
end
end
end || Node <- Nodes
],
logSuccess("Reloaded(Beam changed) Mod: ~s on ~p nodes:~p", [Module, NumNodes, Nodes]).
recompileChangeSrcFile(Iterator, SwSyncNode) ->
case maps:next(Iterator) of
{File, _V, NextIterator} ->
recompileSrcFile(File, SwSyncNode),
recompileChangeSrcFile(NextIterator, SwSyncNode);
_ ->
ok
end.
erlydtlCompile(SrcFile, Options) ->
F =
fun({outdir, OutDir}, Acc) -> [{out_dir, OutDir} | Acc];
(OtherOption, Acc) -> [OtherOption | Acc]
end,
DtlOptions = lists:foldl(F, [], Options),
Module = binary_to_atom(lists:flatten(filename:basename(SrcFile, ".dtl") ++ "_dtl")),
Compiler = erlydtl,
Compiler:compile(SrcFile, Module, DtlOptions).
elixir_compile(SrcFile, Options) ->
Outdir = proplists:get_value(outdir, Options),
Compiler = ':Elixir.Kernel.ParallelCompiler',
Modules = Compiler:files_to_path([SrcFile], Outdir),
Loader =
fun(Module) ->
Outfile = code:which(Module),
Binary = file:read_file(Outfile),
{Module, Binary}
end,
Results = lists:map(Loader, Modules),
{ok, multiple, Results, []}.
lfe_compile(SrcFile, Options) ->
Compiler = lfe_comp,
Compiler:file(SrcFile, Options).
getCompileFunAndModuleName(SrcFile) ->
case getFileType(SrcFile) of
erl ->
{fun compile:file/2, binary_to_atom(filename:basename(SrcFile, <<".erl">>))};
dtl ->
{fun erlydtlCompile/2, list_to_atom(lists:flatten(binary_to_list(filename:basename(SrcFile, <<".dtl">>)) ++ "_dtl"))};
lfe ->
{fun lfe_compile/2, binary_to_atom(filename:basename(SrcFile, <<".lfe">>))};
elixir ->
{fun elixir_compile/2, binary_to_atom(filename:basename(SrcFile, <<".ex">>))}
end.
getObjectCode(Module) ->
case code:get_object_code(Module) of
{Module, B, Filename} -> {B, Filename};
_ -> {undefined, undefined}
end.
reloadIfNecessary(Module, OldBinary, Binary, Filename, SwSyncNode) ->
case Binary =/= OldBinary of
true ->
%% Try to load the module...
case code:ensure_loaded(Module) of
{module, Module} ->
case code:load_binary(Module, Filename, Binary) of
{module, Module} ->
logSuccess("Reloaded(Beam recompiled) Mod:~s Success", [Module]),
syncLoadModOnAllNodes(SwSyncNode, Module, Binary, recompiled);
{error, What} ->
logErrors("Reloaded(Beam recompiled) Mod:~s Errors Reason:~p", [Module, What])
end;
{error, nofile} ->
case code:load_binary(Module, Filename, Binary) of
{module, Module} ->
logSuccess("Reloaded(Beam recompiled) Mod:~s Success", [Module]),
syncLoadModOnAllNodes(SwSyncNode, Module, Binary, recompiled);
{error, What} ->
logErrors("Reloaded(Beam recompiled) Mod:~s Errors Reason:~p", [Module, What])
end;
{error, embedded} ->
case code:load_file(Module) of %% Module is not yet loaded, load it.
{module, Module} -> ok;
{error, nofile} ->
case code:load_binary(Module, Filename, Binary) of
{module, Module} ->
logSuccess("Reloaded(Beam recompiled) Mod:~s Success", [Module]),
syncLoadModOnAllNodes(SwSyncNode, Module, Binary, recompiled);
{error, What} ->
logErrors("Reloaded(Beam recompiled) Mod:~s Errors Reason:~p", [Module, What])
end
end
end;
_ ->
ignore
end.
recompileSrcFile(SrcFile, SwSyncNode) ->
%% Get the module, src dir, and options...
RootSrcDir =
case getRootSrcDirFromSrcDir(SrcFile) of
undefined ->
filename:dirname(SrcFile);
RetSrcDir ->
RetSrcDir
end,
CurSrcDir = filename:dirname(SrcFile),
{CompileFun, Module} = getCompileFunAndModuleName(SrcFile),
{OldBinary, Filename} = getObjectCode(Module),
case Module == es_gen_ipc orelse getOptions(RootSrcDir) of
true ->
ignore;
{ok, Options} ->
RightFileDir = binary_to_list(filename:join(CurSrcDir, filename:basename(SrcFile))),
LastOptions = ?IIF(?esCfgSync:getv(?isJustMem), [binary, return | Options], [return | Options]),
case CompileFun(RightFileDir, LastOptions) of
{ok, Module, Binary, Warnings} ->
printResults(Module, RightFileDir, [], Warnings),
reloadIfNecessary(Module, OldBinary, Binary, Filename, SwSyncNode),
{ok, [], Warnings};
{ok, [{ok, Module, Binary, Warnings}], Warnings2} ->
printResults(Module, RightFileDir, [], Warnings ++ Warnings2),
reloadIfNecessary(Module, OldBinary, Binary, Filename, SwSyncNode),
{ok, [], Warnings ++ Warnings2};
{ok, multiple, Results, Warnings} ->
printResults(Module, RightFileDir, [], Warnings),
[reloadIfNecessary(CompiledModule, OldBinary, Binary, Filename, SwSyncNode) || {CompiledModule, Binary} <- Results],
{ok, [], Warnings};
{ok, OtherModule, _Binary, Warnings} ->
Desc = io_lib:format("Module definition (~p) differs from expected (~s)", [OtherModule, filename:rootname(filename:basename(RightFileDir))]),
Errors = [{RightFileDir, {0, Module, Desc}}],
printResults(Module, RightFileDir, Errors, Warnings),
{ok, Errors, Warnings};
{error, Errors, Warnings} ->
printResults(Module, RightFileDir, Errors, Warnings),
{ok, Errors, Warnings};
{ok, Module, Warnings} ->
printResults(Module, RightFileDir, [], Warnings),
{ok, [], Warnings};
_Err ->
logErrors("compile Mod:~s Errors Reason:~p", [Module, _Err])
end;
undefined ->
case tryGetModOpts(Module) of
{ok, Options} ->
setOptions(RootSrcDir, Options),
recompileSrcFile(SrcFile, SwSyncNode);
_ ->
case tryGetSrcOpts(CurSrcDir) of
{ok, Options} ->
setOptions(RootSrcDir, Options),
recompileSrcFile(SrcFile, SwSyncNode);
_ ->
logErrors("Unable to determine options for ~s", [SrcFile])
end
end
end.
collIncludeCHrls([], AllHrls, CHrls, NewAddMap) ->
case maps:size(NewAddMap) > 0 of
true ->
collIncludeCHrls(maps:keys(NewAddMap), AllHrls, CHrls, #{});
_ ->
CHrls
end;
collIncludeCHrls([OneHrl | LeftCHrls], AllHrls, CHrls, NewAddMap) ->
{NewCHrls, NNewAddMap} = whoInclude(OneHrl, AllHrls, CHrls, NewAddMap),
collIncludeCHrls(LeftCHrls, AllHrls, NewCHrls, NNewAddMap).
collIncludeCErls([], _SrcFiles, CSrcs, _NewAddMap) ->
CSrcs;
collIncludeCErls([Hrl | LeftHrl], SrcFiles, CSrcs, NewAddMap) ->
{NewCSrcs, NNewAddMap} = whoInclude(Hrl, SrcFiles, CSrcs, NewAddMap),
collIncludeCErls(LeftHrl, SrcFiles, NewCSrcs, NNewAddMap).
whoInclude(HrlFile, AllFiles, CFiles, NewAddMap) ->
HrlFileBaseName = filename:basename(HrlFile),
QuoteHrlFileBaseName = <<"\"", HrlFileBaseName/binary, "\"">>,
doMathEveryFile(maps:iterator(AllFiles), QuoteHrlFileBaseName, CFiles, NewAddMap).
doMathEveryFile(Iterator, HrlFileBaseName, CFiles, NewAddMap) ->
case maps:next(Iterator) of
{OneFile, _V, NextIterator} ->
case file:open(OneFile, [read, binary]) of
{ok, IoDevice} ->
IsInclude = doMathEveryLine(IoDevice, HrlFileBaseName),
file:close(IoDevice),
case IsInclude of
true ->
case maps:is_key(OneFile, CFiles) of
true ->
doMathEveryFile(NextIterator, HrlFileBaseName, CFiles, NewAddMap);
_ ->
doMathEveryFile(NextIterator, HrlFileBaseName, CFiles#{OneFile => 1}, NewAddMap#{OneFile => 1})
end;
_ ->
doMathEveryFile(NextIterator, HrlFileBaseName, CFiles, NewAddMap)
end;
_ ->
doMathEveryFile(NextIterator, HrlFileBaseName, CFiles, NewAddMap)
end;
_ ->
{CFiles, NewAddMap}
end.
%% 注释
%% whoInclude(HrlFile, SrcFiles) ->
%% HrlFileBaseName = filename:basename(HrlFile),
%% Pred =
%% fun(SrcFile, _) ->
%% {ok, Forms} = epp_dodger:parse_file(SrcFile),
%% isInclude(binary_to_list(HrlFileBaseName), Forms)
%% end,
%% maps:filter(Pred, SrcFiles).
%% isInclude(_HrlFile, []) ->
%% false;
%% isInclude(HrlFile, [{tree, attribute, _, {attribute, _, [{_, _, IncludeFile}]}} | Forms]) when is_list(IncludeFile) ->
%% IncludeFileBaseName = filename:basename(IncludeFile),
%% case IncludeFileBaseName of
%% HrlFile -> true;
%% _ -> isInclude(HrlFile, Forms)
%% end;
%% isInclude(HrlFile, [_SomeForm | Forms]) ->
%% isInclude(HrlFile, Forms).
doMathEveryLine(IoDevice, HrlFileBaseName) ->
case file:read_line(IoDevice) of
{ok, Data} ->
case re:run(Data, HrlFileBaseName) of
nomatch ->
case re:run(Data, <<"->">>) of
nomatch ->
doMathEveryLine(IoDevice, HrlFileBaseName);
_ ->
false
end;
_ ->
true
end;
_ ->
false
end.
classifyChangeFile([], Beams, Configs, Hrls, Srcs, ColSrcs, ColHrls, ColConfigs, ColBeams) ->
{Beams, Configs, Hrls, Srcs, ColSrcs, ColHrls, ColConfigs, ColBeams};
classifyChangeFile([OneFile | LeftFile], Beams, Configs, Hrls, Srcs, ColSrcs, ColHrls, ColConfigs, ColBeams) ->
CurMTimeSec = dateTimeToSec(filelib:last_modified(OneFile)),
case filename:extension(OneFile) of
<<".beam">> ->
BinMod = binary_to_atom(filename:basename(OneFile, <<".beam">>)),
case ColBeams of
#{BinMod := OldMTimeSec} ->
case CurMTimeSec =/= OldMTimeSec andalso CurMTimeSec =/= 0 of
true ->
classifyChangeFile(LeftFile, [BinMod | Beams], Configs, Hrls, Srcs, ColSrcs, ColHrls, ColConfigs, ColBeams#{BinMod := CurMTimeSec});
_ ->
classifyChangeFile(LeftFile, Beams, Configs, Hrls, Srcs, ColSrcs, ColHrls, ColConfigs, ColBeams)
end;
_ ->
classifyChangeFile(LeftFile, [BinMod | Beams], Configs, Hrls, Srcs, ColSrcs, ColHrls, ColConfigs, ColBeams#{BinMod => CurMTimeSec})
end;
<<".config">> ->
AbsFile = filename:absname(OneFile),
case ColConfigs of
#{AbsFile := OldMTimeSec} ->
case CurMTimeSec =/= OldMTimeSec andalso CurMTimeSec =/= 0 of
true ->
CfgMod = erlang:binary_to_atom(filename:basename(AbsFile, <<".config">>), utf8),
classifyChangeFile(LeftFile, Beams, [CfgMod | Configs], Hrls, Srcs, ColSrcs, ColHrls, ColConfigs#{AbsFile := CurMTimeSec}, ColBeams);
_ ->
classifyChangeFile(LeftFile, Beams, Configs, Hrls, Srcs, ColSrcs, ColHrls, ColConfigs, ColBeams)
end;
_ ->
CfgMod = erlang:binary_to_atom(filename:basename(AbsFile, <<".config">>), utf8),
classifyChangeFile(LeftFile, Beams, [CfgMod | Configs], Hrls, Srcs, ColSrcs, ColHrls, ColConfigs#{AbsFile => CurMTimeSec}, ColBeams)
end;
<<".hrl">> ->
AbsFile = filename:absname(OneFile),
case ColHrls of
#{AbsFile := OldMTimeSec} ->
case CurMTimeSec =/= OldMTimeSec andalso CurMTimeSec =/= 0 of
true ->
classifyChangeFile(LeftFile, Beams, Configs, Hrls#{AbsFile => 1}, Srcs, ColSrcs, ColHrls#{AbsFile := CurMTimeSec}, ColConfigs, ColBeams);
_ ->
classifyChangeFile(LeftFile, Beams, Configs, Hrls, Srcs, ColSrcs, ColHrls, ColConfigs, ColBeams)
end;
_ ->
classifyChangeFile(LeftFile, Beams, Configs, Hrls#{AbsFile => 1}, Srcs, ColSrcs, ColHrls#{AbsFile => CurMTimeSec}, ColConfigs, ColBeams)
end;
<<>> ->
classifyChangeFile(LeftFile, Beams, Configs, Hrls, Srcs, ColSrcs, ColHrls, ColConfigs, ColBeams);
_ ->
AbsFile = filename:absname(OneFile),
case ColSrcs of
#{AbsFile := OldMTimeSec} ->
case CurMTimeSec =/= OldMTimeSec andalso CurMTimeSec =/= 0 of
true ->
classifyChangeFile(LeftFile, Beams, Configs, Hrls, Srcs#{AbsFile => 1}, ColSrcs#{AbsFile := CurMTimeSec}, ColHrls, ColConfigs, ColBeams);
_ ->
classifyChangeFile(LeftFile, Beams, Configs, Hrls, Srcs, ColSrcs, ColHrls, ColConfigs, ColBeams)
end;
_ ->
classifyChangeFile(LeftFile, Beams, Configs, Hrls, Srcs#{AbsFile => 1}, ColSrcs#{AbsFile => CurMTimeSec}, ColHrls, ColConfigs, ColBeams)
end
end.
fileSyncPath(ExecName) ->
case code:priv_dir(?MODULE) of
{error, _} ->
case code:which(?MODULE) of
Filename when is_list(Filename) ->
filename:join([filename:dirname(filename:dirname(Filename)), "priv", ExecName]);
_ ->
filename:join("../priv", ExecName)
end;
Dir ->
filename:join(Dir, ExecName)
end.
dateTimeToSec(DateTime) ->
if
DateTime == 0 ->
0;
true ->
erlang:universaltime_to_posixtime(DateTime)
end.
debugInfoKeyFun() ->
case ?esCfgSync:getv(?debugInfoKeyFun) of
undefined ->
{debug_info_key, undefined};
{Mod, Fun} ->
Mod:Fun()
end.
%% ************************************************* utils end *******************************************************