mirror of
https://github.com/correl/rebar.git
synced 2025-01-07 03:00:15 +00:00
Overhaul environment expansion for better performance
The introduction of setup_env as a global concept caused the rebar_port_compiler implementation to start getting called a LOT. The expansion of environment variables that happens in the port compiler was O(n^n), which means you could see upwards of 80k invocations of lists:foldl on a single app "./rebar clean". This commit reworks the expansion to be O(n^2), and reduces the running time for the same operation by 60%+. On a large project like Riak, the end result is that a build went from 200 seconds to 73.
This commit is contained in:
parent
b65d5904f5
commit
17616d1078
2 changed files with 53 additions and 34 deletions
|
@ -290,38 +290,51 @@ merge_each_var([{Key, Value} | Rest], Vars) ->
|
||||||
%% for every other key until no further expansions are possible.
|
%% for every other key until no further expansions are possible.
|
||||||
%%
|
%%
|
||||||
expand_vars_loop(Vars) ->
|
expand_vars_loop(Vars) ->
|
||||||
expand_vars_loop(Vars, 10).
|
expand_vars_loop(Vars, [], dict:from_list(Vars), 10).
|
||||||
|
|
||||||
expand_vars_loop(_, 0) ->
|
expand_vars_loop(_Pending, _Recurse, _Vars, 0) ->
|
||||||
?ABORT("Max. expansion reached for ENV vars!\n", []);
|
?ABORT("Max. expansion reached for ENV vars!\n", []);
|
||||||
expand_vars_loop(Vars0, Count) ->
|
expand_vars_loop([], [], Vars, _Count) ->
|
||||||
Vars = lists:foldl(fun({Key, Value}, Acc) ->
|
lists:keysort(1, dict:to_list(Vars));
|
||||||
expand_vars(Key, Value, Acc)
|
expand_vars_loop([], Recurse, Vars, Count) ->
|
||||||
end,
|
expand_vars_loop(Recurse, [], Vars, Count-1);
|
||||||
Vars0, Vars0),
|
expand_vars_loop([{K, V} | Rest], Recurse, Vars, Count) ->
|
||||||
case orddict:from_list(Vars) of
|
%% Identify the variables that need expansion in this value
|
||||||
Vars0 ->
|
case re:run(V, "\\\${?(\\w+)}?", [global, {capture, all_but_first, list}]) of
|
||||||
Vars0;
|
{match, Matches} ->
|
||||||
_ ->
|
%% Identify the unique variables that need to be expanded
|
||||||
expand_vars_loop(Vars, Count-1)
|
UniqueMatches = lists:usort([M || [M] <- Matches]),
|
||||||
|
|
||||||
|
%% For each variable, expand it and return the final value. Note that
|
||||||
|
%% if we have a bunch of unresolvable variables, nothing happens and
|
||||||
|
%% we don't bother attempting further expansion
|
||||||
|
case expand_keys_in_value(UniqueMatches, V, Vars) of
|
||||||
|
V ->
|
||||||
|
%% No change after expansion; move along
|
||||||
|
expand_vars_loop(Rest, Recurse, Vars, Count);
|
||||||
|
Expanded ->
|
||||||
|
%% Some expansion occurred; move to next k/v but revist
|
||||||
|
%% this value in the next loop to check for further expansion
|
||||||
|
NewVars = dict:store(K, Expanded, Vars),
|
||||||
|
expand_vars_loop(Rest, [{K, Expanded} | Recurse], NewVars, Count)
|
||||||
|
end;
|
||||||
|
|
||||||
|
nomatch ->
|
||||||
|
%% No values in this variable need expansion; move along
|
||||||
|
expand_vars_loop(Rest, Recurse, Vars, Count)
|
||||||
end.
|
end.
|
||||||
|
|
||||||
%%
|
expand_keys_in_value([], Value, _Vars) ->
|
||||||
%% Expand all OTHER references to a given K/V pair
|
Value;
|
||||||
%%
|
expand_keys_in_value([Key | Rest], Value, Vars) ->
|
||||||
expand_vars(Key, Value, Vars) ->
|
NewValue = case dict:find(Key, Vars) of
|
||||||
lists:foldl(
|
{ok, KValue} ->
|
||||||
fun({AKey, AValue}, Acc) ->
|
rebar_utils:expand_env_variable(Value, Key, KValue);
|
||||||
NewValue = case AKey of
|
error ->
|
||||||
Key ->
|
Value
|
||||||
AValue;
|
|
||||||
_ ->
|
|
||||||
rebar_utils:expand_env_variable(AValue,
|
|
||||||
Key, Value)
|
|
||||||
end,
|
end,
|
||||||
[{AKey, NewValue} | Acc]
|
expand_keys_in_value(Rest, NewValue, Vars).
|
||||||
end,
|
|
||||||
[], Vars).
|
|
||||||
|
|
||||||
expand_command(TmplName, Env, InFiles, OutFile) ->
|
expand_command(TmplName, Env, InFiles, OutFile) ->
|
||||||
Cmd0 = proplists:get_value(TmplName, Env),
|
Cmd0 = proplists:get_value(TmplName, Env),
|
||||||
|
|
|
@ -186,12 +186,18 @@ expand_code_path() ->
|
||||||
%% The end of form $FOO is delimited with whitespace or eol
|
%% The end of form $FOO is delimited with whitespace or eol
|
||||||
%%
|
%%
|
||||||
expand_env_variable(InStr, VarName, RawVarValue) ->
|
expand_env_variable(InStr, VarName, RawVarValue) ->
|
||||||
|
case string:chr(InStr, $$) of
|
||||||
|
0 ->
|
||||||
|
%% No variables to expand
|
||||||
|
InStr;
|
||||||
|
_ ->
|
||||||
ReOpts = [global, {return, list}],
|
ReOpts = [global, {return, list}],
|
||||||
VarValue = re:replace(RawVarValue, "\\\\", "\\\\\\\\", ReOpts),
|
VarValue = re:replace(RawVarValue, "\\\\", "\\\\\\\\", [global]),
|
||||||
R1 = re:replace(InStr, "\\\$" ++ VarName ++ "\\s", VarValue ++ " ",
|
%% Use a regex to match/replace:
|
||||||
[global]),
|
%% $FOO\s | ${FOO} | $FOO eol
|
||||||
R2 = re:replace(R1, "\\\$" ++ VarName ++ "\$", VarValue),
|
RegEx = io_lib:format("\\\$(~s(\\s|$)|{~s})", [VarName, VarName]),
|
||||||
re:replace(R2, "\\\${" ++ VarName ++ "}", VarValue, ReOpts).
|
re:replace(InStr, RegEx, VarValue ++ "\\2", ReOpts)
|
||||||
|
end.
|
||||||
|
|
||||||
|
|
||||||
%% ====================================================================
|
%% ====================================================================
|
||||||
|
|
Loading…
Reference in a new issue