From 6cfe28c954dc12f847a15a78b5fcf65154f0407a Mon Sep 17 00:00:00 2001 From: Fred Hebert Date: Sat, 6 Dec 2014 16:29:44 +0000 Subject: Partial fix to circular deps (#40) - Adding tests - fixing use of set fetching to find repeated deps and prevent infinite loops On a circular loop rebar3 now fails with `{error, no_sort}`, which is uncaught and should be handled to consider the issue fully fixed. --- src/rebar_prv_install_deps.erl | 6 ++--- test/mock_git_resource.erl | 2 +- test/rebar_compile_SUITE.erl | 2 +- test/rebar_deps_SUITE.erl | 35 ++++++++++++++++++++-------- test/rebar_test_utils.erl | 52 +++++++++++++++++++++++++----------------- 5 files changed, 62 insertions(+), 35 deletions(-) diff --git a/src/rebar_prv_install_deps.erl b/src/rebar_prv_install_deps.erl index a3ffc66..461baa5 100644 --- a/src/rebar_prv_install_deps.erl +++ b/src/rebar_prv_install_deps.erl @@ -151,7 +151,7 @@ update_pkg_deps(Pkgs, Packages, Update, Seen, State) -> ,Packages ,Pkg), {SeenAcc1, StateAcc1} = maybe_lock(AppInfo, SeenAcc, StateAcc), - case maybe_fetch(StateAcc1, AppInfo, Update, SeenAcc) of + case maybe_fetch(StateAcc1, AppInfo, Update, SeenAcc1) of true -> {[AppInfo | Acc], SeenAcc1, StateAcc1}; false -> @@ -167,7 +167,7 @@ maybe_lock(AppInfo, Seen, State) -> {sets:add_element(Name, Seen), rebar_state:lock(State, AppInfo)}; true -> - {sets:add_element(Name, Seen), State} + {Seen, State} end. package_to_app(DepsDir, Packages, {Name, Vsn}) -> @@ -204,7 +204,7 @@ update_src_deps(Level, SrcDeps, PkgDeps, SrcApps, State, Update, Seen) -> ,Level ,StateAcc1); _ -> - maybe_fetch(StateAcc, AppInfo, false, SeenAcc), + maybe_fetch(StateAcc, AppInfo, false, SeenAcc1), handle_dep(AppInfo ,SrcDepsAcc ,PkgDepsAcc diff --git a/test/mock_git_resource.erl b/test/mock_git_resource.erl index 8e50ec4..d867a28 100644 --- a/test/mock_git_resource.erl +++ b/test/mock_git_resource.erl @@ -102,8 +102,8 @@ mock_download(Opts) -> meck:expect( ?MOD, download, fun (Dir, Git) -> - {git, Url, {_, Vsn}} = normalize_git(Git, Overrides, Default), filelib:ensure_dir(Dir), + {git, Url, {_, Vsn}} = normalize_git(Git, Overrides, Default), App = app(Url), AppDeps = proplists:get_value(App, Deps, []), rebar_test_utils:create_app( diff --git a/test/rebar_compile_SUITE.erl b/test/rebar_compile_SUITE.erl index 6c5cd1e..83b868d 100644 --- a/test/rebar_compile_SUITE.erl +++ b/test/rebar_compile_SUITE.erl @@ -33,5 +33,5 @@ build_basic_app(Config) -> Vsn = rebar_test_utils:create_random_vsn(), rebar_test_utils:create_app(AppDir, Name, Vsn, [kernel, stdlib]), - rebar_test_utils:run_and_check(Config, [], "compile", [{app, Name}]). + rebar_test_utils:run_and_check(Config, [], "compile", {ok, [{app, Name}]}). diff --git a/test/rebar_deps_SUITE.erl b/test/rebar_deps_SUITE.erl index b6ee8b8..6e5b874 100644 --- a/test/rebar_deps_SUITE.erl +++ b/test/rebar_deps_SUITE.erl @@ -4,7 +4,8 @@ -include_lib("common_test/include/ct.hrl"). -include_lib("eunit/include/eunit.hrl"). -all() -> [flat, pick_highest_left, pick_highest_right, pick_earliest]. +all() -> [flat, pick_highest_left, pick_highest_right, pick_earliest, + circular1, circular2]. init_per_suite(Config) -> application:start(meck), @@ -15,29 +16,42 @@ end_per_suite(_Config) -> init_per_testcase(Case, Config) -> {Deps, Expect} = deps(Case), - [{expect, - [case Dep of + Expected = case Expect of + {ok, List} -> {ok, format_expected_deps(List)}; + {error, Reason} -> {error, Reason} + end, + [{expect, Expected} | setup_project(Case, Config, expand_deps(Deps))]. + +format_expected_deps(Deps) -> + [case Dep of {N,V} -> {dep, N, V}; N -> {dep, N} - end || Dep <- Expect]} - | setup_project(Case, Config, expand_deps(Deps))]. + end || Dep <- Deps]. deps(flat) -> {[{"B", []}, {"C", []}], - ["B", "C"]}; + {ok, ["B", "C"]}}; deps(pick_highest_left) -> {[{"B", [{"C", "2", []}]}, {"C", "1", []}], - ["B", {"C", "1"}]}; % Warn C2 + {ok, ["B", {"C", "1"}]}}; % Warn C2 deps(pick_highest_right) -> {[{"B", "1", []}, {"C", [{"B", "2", []}]}], - [{"B","1"}, "C"]}; % Warn B2 + {ok, [{"B","1"}, "C"]}}; % Warn B2 deps(pick_earliest) -> {[{"B", [{"D", "1", []}]}, {"C", [{"D", "2", []}]}], - ["B","C",{"D","1"}]}. % Warn D2 + {ok, ["B","C",{"D","1"}]}}; % Warn D2 +deps(circular1) -> + {[{"B", [{"A", []}]}, % A is the top-level app + {"C", []}], + {error, no_sort}}; % circular dep +deps(circular2) -> + {[{"B", [{"C", [{"B", []}]}]}, + {"C", []}], + {error, no_sort}}. % circular dep end_per_testcase(_, Config) -> mock_git_resource:unmock(), @@ -55,6 +69,7 @@ expand_deps([{Name, Vsn, Deps} | Rest]) -> setup_project(Case, Config0, Deps) -> Config = rebar_test_utils:init_rebar_state(Config0, atom_to_list(Case)), AppDir = ?config(apps, Config), + rebar_test_utils:create_app(AppDir, "A", "0.0.0", [kernel, stdlib]), TopDeps = top_level_deps(Deps), RebarConf = rebar_test_utils:create_config(AppDir, [{deps, TopDeps}]), mock_git_resource:mock([{deps, flat_deps(Deps)}]), @@ -76,6 +91,8 @@ flat(Config) -> run(Config). pick_highest_left(Config) -> run(Config). pick_highest_right(Config) -> run(Config). pick_earliest(Config) -> run(Config). +circular1(Config) -> run(Config). +circular2(Config) -> run(Config). run(Config) -> {ok, RebarConfig} = file:consult(?config(rebarconfig, Config)), diff --git a/test/rebar_test_utils.erl b/test/rebar_test_utils.erl index 08834f7..ed494a5 100644 --- a/test/rebar_test_utils.erl +++ b/test/rebar_test_utils.erl @@ -45,27 +45,14 @@ run_and_check(Config, RebarConfig, Command, Expect) -> %% Assumes init_rebar_state has run first AppDir = ?config(apps, Config), State = ?config(state, Config), - {ok,_} = rebar3:run(rebar_state:new(State, RebarConfig, AppDir), Command), - BuildDir = filename:join([AppDir, "_build", "default", "lib"]), - Deps = rebar_app_discover:find_apps([BuildDir], all), - DepsNames = [{ec_cnv:to_list(rebar_app_info:name(App)), App} || App <- Deps], - lists:foreach( - fun({app, Name}) -> - [App] = rebar_app_discover:find_apps([AppDir]), - ct:pal("Name: ~p", [Name]), - ?assertEqual(Name, ec_cnv:to_list(rebar_app_info:name(App))) - ; ({dep, Name}) -> - ct:pal("Name: ~p", [Name]), - ?assertNotEqual(false, lists:keyfind(Name, 1, DepsNames)) - ; ({dep, Name, Vsn}) -> - ct:pal("Name: ~p, Vsn: ~p", [Name, Vsn]), - case lists:keyfind(Name, 1, DepsNames) of - false -> - error({app_not_found, Name}); - {Name, App} -> - ?assertEqual(Vsn, rebar_app_info:original_vsn(App)) - end - end, Expect). + Res = rebar3:run(rebar_state:new(State, RebarConfig, AppDir), Command), + case Expect of + {error, Reason} -> + ?assertEqual({error, Reason}, Res); + {ok, Expected} -> + {ok, _} = Res, + check_results(AppDir, Expected) + end. %% @doc Creates a dummy application including: %% - src/.erl @@ -108,6 +95,28 @@ create_random_vsn() -> %%%%%%%%%%%%%%% %%% Helpers %%% %%%%%%%%%%%%%%% +check_results(AppDir, Expected) -> + BuildDir = filename:join([AppDir, "_build", "default", "lib"]), + Deps = rebar_app_discover:find_apps([BuildDir], all), + DepsNames = [{ec_cnv:to_list(rebar_app_info:name(App)), App} || App <- Deps], + lists:foreach( + fun({app, Name}) -> + [App] = rebar_app_discover:find_apps([AppDir]), + ct:pal("Name: ~p", [Name]), + ?assertEqual(Name, ec_cnv:to_list(rebar_app_info:name(App))) + ; ({dep, Name}) -> + ct:pal("Name: ~p", [Name]), + ?assertNotEqual(false, lists:keyfind(Name, 1, DepsNames)) + ; ({dep, Name, Vsn}) -> + ct:pal("Name: ~p, Vsn: ~p", [Name, Vsn]), + case lists:keyfind(Name, 1, DepsNames) of + false -> + error({app_not_found, Name}); + {Name, App} -> + ?assertEqual(Vsn, rebar_app_info:original_vsn(App)) + end + end, Expected). + write_src_file(Dir, Name) -> Erl = filename:join([Dir, "src", "not_a_real_src" ++ Name ++ ".erl"]), ok = filelib:ensure_dir(Erl), @@ -136,3 +145,4 @@ get_app_metadata(Name, Vsn, Deps) -> {included_applications, []}, {registered, []}, {applications, Deps}]}. + -- cgit v1.1 From 63003c3986ca20e2f733e79453315fa0b7076bd9 Mon Sep 17 00:00:00 2001 From: Fred Hebert Date: Sat, 6 Dec 2014 23:18:35 +0000 Subject: Return cycles in deps solver --- src/rebar_digraph.erl | 10 +++++++++- test/rebar_deps_SUITE.erl | 4 ++-- 2 files changed, 11 insertions(+), 3 deletions(-) diff --git a/src/rebar_digraph.erl b/src/rebar_digraph.erl index bb031cb..dbcf649 100644 --- a/src/rebar_digraph.erl +++ b/src/rebar_digraph.erl @@ -18,7 +18,15 @@ compile_order(Apps) -> end, Apps), case digraph_utils:topsort(Graph) of false -> - {error, no_sort}; + case digraph_utils:is_acyclic(Graph) of + true -> + {error, no_sort}; + false -> + Cycles = lists:sort( + [lists:sort(Comp) || Comp <- digraph_utils:strong_components(Graph), + length(Comp)>1]), + {error, {cycles, Cycles}} + end; V -> {ok, names_to_apps(lists:reverse(V), Apps)} end. diff --git a/test/rebar_deps_SUITE.erl b/test/rebar_deps_SUITE.erl index 6e5b874..a98f690 100644 --- a/test/rebar_deps_SUITE.erl +++ b/test/rebar_deps_SUITE.erl @@ -47,11 +47,11 @@ deps(pick_earliest) -> deps(circular1) -> {[{"B", [{"A", []}]}, % A is the top-level app {"C", []}], - {error, no_sort}}; % circular dep + {error, {cycles, [[<<"A">>,<<"B">>]]}}}; % circular dep deps(circular2) -> {[{"B", [{"C", [{"B", []}]}]}, {"C", []}], - {error, no_sort}}. % circular dep + {error, {cycles, [[<<"B">>,<<"C">>]]}}}. % circular dep end_per_testcase(_, Config) -> mock_git_resource:unmock(), -- cgit v1.1 From be39ab9e7bd1553132def2a1e89f1ce38f6f2fcc Mon Sep 17 00:00:00 2001 From: Fred Hebert Date: Sat, 6 Dec 2014 23:53:11 +0000 Subject: Handle cycle errors in provider --- src/rebar_prv_install_deps.erl | 8 ++++++++ test/rebar_deps_SUITE.erl | 4 ++-- 2 files changed, 10 insertions(+), 2 deletions(-) diff --git a/src/rebar_prv_install_deps.erl b/src/rebar_prv_install_deps.erl index 461baa5..1be094c 100644 --- a/src/rebar_prv_install_deps.erl +++ b/src/rebar_prv_install_deps.erl @@ -81,6 +81,8 @@ do(State) -> {ok, rebar_state:deps_to_build(State1, lists:dropwhile(fun rebar_app_info:valid/1 , Sort -- ProjectApps))}; + {error, {cycles, Cycles}} -> + {error, {?MODULE, {cycles, Cycles}}}; {error, Error} -> {error, Error} end @@ -91,6 +93,12 @@ do(State) -> end. -spec format_error(any()) -> iolist(). +format_error({cycles, Cycles}) -> + Prints = [["applications: ", + [io_lib:format("~s ", [Dep]) || Dep <- Cycle], + "depend on each other~n"] + || Cycle <- Cycles], + ["Dependency cycle(s) detected:~n", Prints]; format_error(Reason) -> io_lib:format("~p", [Reason]). diff --git a/test/rebar_deps_SUITE.erl b/test/rebar_deps_SUITE.erl index a98f690..656c3b5 100644 --- a/test/rebar_deps_SUITE.erl +++ b/test/rebar_deps_SUITE.erl @@ -47,11 +47,11 @@ deps(pick_earliest) -> deps(circular1) -> {[{"B", [{"A", []}]}, % A is the top-level app {"C", []}], - {error, {cycles, [[<<"A">>,<<"B">>]]}}}; % circular dep + {error, {rebar_prv_install_deps, {cycles, [[<<"A">>,<<"B">>]]}}}}; deps(circular2) -> {[{"B", [{"C", [{"B", []}]}]}, {"C", []}], - {error, {cycles, [[<<"B">>,<<"C">>]]}}}. % circular dep + {error, {rebar_prv_install_deps, {cycles, [[<<"B">>,<<"C">>]]}}}}. end_per_testcase(_, Config) -> mock_git_resource:unmock(), -- cgit v1.1