summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSiri Hansen <siri@erlang.org>2016-12-16 10:34:36 +0100
committerSiri Hansen <siri@erlang.org>2016-12-20 09:18:19 +0100
commit4e52ce58e505ba57f2188e34a002d70de1bd6ff5 (patch)
treeaaa13fdf8b63b2be8d76d5a1f9ff37e53304bc5a
parenta18340c6eed6e62237c4ec63f15daacfe86dc86c (diff)
Add all dirs from test spec
Parse given test specs and add all spec- and suite directories as extra_src_dirs in order to ensure that all these directories are copied to the _build area and the suites are compiled. Specs located in the project- or app root are explicitly copied to the _build area in order to avoid recursive copying of the complete directory tree.
-rw-r--r--src/rebar_prv_common_test.erl81
-rw-r--r--test/rebar_ct_SUITE.erl103
2 files changed, 168 insertions, 16 deletions
diff --git a/src/rebar_prv_common_test.erl b/src/rebar_prv_common_test.erl
index 9db8106..4a9ba25 100644
--- a/src/rebar_prv_common_test.erl
+++ b/src/rebar_prv_common_test.erl
@@ -429,9 +429,18 @@ test_dirs(State, Apps, Opts) ->
set_compile_dirs(State, Apps, join(Suites, Dir));
{_Suites, _Dirs} -> {error, "Only a single directory may be specified when specifying suites"}
end;
- Specs ->
- set_compile_dirs(State, Apps, {spec, Specs})
-
+ Specs0 ->
+ case get_dirs_from_specs(Specs0) of
+ {ok,{Specs,SuiteDirs}} ->
+ {State1,Apps1} = set_compile_dirs1(State, Apps,
+ {dir, SuiteDirs}),
+ {State2,Apps2} = set_compile_dirs1(State1, Apps1,
+ {spec, Specs}),
+ [maybe_copy_spec(State2,Apps2,S) || S <- Specs],
+ {ok, rebar_state:project_apps(State2, Apps2)};
+ Error ->
+ Error
+ end
end.
join(Suite, Dir) when is_integer(hd(Suite)) ->
@@ -439,24 +448,25 @@ join(Suite, Dir) when is_integer(hd(Suite)) ->
join(Suites, Dir) ->
{suite, lists:map(fun(S) -> filename:join([Dir, S]) end, Suites)}.
-set_compile_dirs(State, Apps, {dir, Dir}) when is_integer(hd(Dir)) ->
+set_compile_dirs(State, Apps, What) ->
+ {NewState,NewApps} = set_compile_dirs1(State, Apps, What),
+ {ok, rebar_state:project_apps(NewState, NewApps)}.
+
+set_compile_dirs1(State, Apps, {dir, Dir}) when is_integer(hd(Dir)) ->
%% single directory
%% insert `Dir` into an app if relative, or the base state if not
%% app relative but relative to the root or not at all if outside
%% project scope
- {NewState, NewApps} = maybe_inject_test_dir(State, [], Apps, Dir),
- {ok, rebar_state:project_apps(NewState, NewApps)};
-set_compile_dirs(State, Apps, {dir, Dirs}) ->
+ maybe_inject_test_dir(State, [], Apps, Dir);
+set_compile_dirs1(State, Apps, {dir, Dirs}) ->
%% multiple directories
F = fun(Dir, {S, A}) -> maybe_inject_test_dir(S, [], A, Dir) end,
- {NewState, NewApps} = lists:foldl(F, {State, Apps}, Dirs),
- {ok, rebar_state:project_apps(NewState, NewApps)};
-set_compile_dirs(State, Apps, {Type, Files}) when Type==spec; Type==suite ->
+ lists:foldl(F, {State, Apps}, Dirs);
+set_compile_dirs1(State, Apps, {Type, Files}) when Type==spec; Type==suite ->
%% specs or suites with dir component
Dirs = find_file_dirs(Files),
F = fun(Dir, {S, A}) -> maybe_inject_test_dir(S, [], A, Dir) end,
- {NewState, NewApps} = lists:foldl(F, {State, Apps}, Dirs),
- {ok, rebar_state:project_apps(NewState, NewApps)}.
+ lists:foldl(F, {State, Apps}, Dirs).
find_file_dirs(Files) ->
AllDirs = lists:map(fun(F) -> filename:dirname(filename:absname(F)) end, Files),
@@ -507,11 +517,58 @@ copy_bare_suites(From, To) ->
ok = rebar_file_utils:cp_r(SrcFiles, To),
rebar_file_utils:cp_r(DataDirs, To).
+maybe_copy_spec(State, [App|Apps], Spec) ->
+ case rebar_file_utils:path_from_ancestor(filename:dirname(Spec), rebar_app_info:dir(App)) of
+ {ok, []} ->
+ ok = rebar_file_utils:cp_r([Spec],rebar_app_info:out_dir(App));
+ {ok,_} ->
+ ok;
+ {error,badparent} ->
+ maybe_copy_spec(State, Apps, Spec)
+ end;
+maybe_copy_spec(State, [], Spec) ->
+ case rebar_file_utils:path_from_ancestor(filename:dirname(Spec), rebar_state:dir(State)) of
+ {ok, []} ->
+ ExtrasDir = filename:join([rebar_dir:base_dir(State), "extras"]),
+ ok = rebar_file_utils:cp_r([Spec],ExtrasDir);
+ _R ->
+ ok
+ end.
+
inject_test_dir(Opts, Dir) ->
%% append specified test targets to app defined `extra_src_dirs`
ExtraSrcDirs = rebar_opts:get(Opts, extra_src_dirs, []),
rebar_opts:set(Opts, extra_src_dirs, ExtraSrcDirs ++ [Dir]).
+get_dirs_from_specs(Specs) ->
+ case get_tests_from_specs(Specs) of
+ {ok,Tests} ->
+ {SpecLists,NodeRunSkipLists} = lists:unzip(Tests),
+ SpecList = lists:append(SpecLists),
+ NodeRunSkipList = lists:append(NodeRunSkipLists),
+ RunList = lists:append([R || {_,R,_} <- NodeRunSkipList]),
+ DirList = [element(1,R) || R <- RunList],
+ {ok,{SpecList,DirList}};
+ Error ->
+ Error
+ end.
+
+get_tests_from_specs(Specs) ->
+ case erlang:function_exported(ct_testspec,get_tests,1) of
+ true ->
+ ct_testspec:get_tests(Specs);
+ false ->
+ case ct_testspec:collect_tests_from_file(Specs,true) of
+ Tests when is_list(Tests) ->
+ {ok,[{S,ct_testspec:prepare_tests(R)} || {S,R} <- Tests]};
+ R when is_tuple(R), element(1,R)==testspec ->
+ %% R15
+ {ok,[{Specs,ct_testspec:prepare_tests(R)}]};
+ Error ->
+ Error
+ end
+ end.
+
translate_paths(State, Opts) ->
case proplists:get_value(spec, Opts) of
undefined ->
diff --git a/test/rebar_ct_SUITE.erl b/test/rebar_ct_SUITE.erl
index a47ea58..91324d2 100644
--- a/test/rebar_ct_SUITE.erl
+++ b/test/rebar_ct_SUITE.erl
@@ -53,6 +53,7 @@
misspecified_ct_compile_opts/1,
misspecified_ct_first_files/1,
testspec/1,
+ testspec_at_root/1,
cmd_vs_cfg_opts/1]).
-include_lib("eunit/include/eunit.hrl").
@@ -71,6 +72,7 @@ all() -> [{group, basic_app},
misspecified_ct_compile_opts,
misspecified_ct_first_files,
testspec,
+ testspec_at_root,
cmd_vs_cfg_opts].
groups() -> [{basic_app, [], [basic_app_default_dirs,
@@ -1302,7 +1304,9 @@ testspec(Config) ->
ok = file:write_file(Spec1, "{suites,\".\",all}.\n"),
Spec2 = filename:join([AppDir, "specs", "another.spec"]),
ok = filelib:ensure_dir(Spec2),
- ok = file:write_file(Spec2, "{suites,\"../test/\",all}.\n"),
+ Suites2 = filename:join([AppDir,"suites","*"]),
+ ok = filelib:ensure_dir(Suites2),
+ ok = file:write_file(Spec2, "{suites,\"../suites/\",all}.\n"),
{ok,Wd} = file:get_cwd(),
ok = file:set_cwd(AppDir),
@@ -1325,7 +1329,7 @@ testspec(Config) ->
{ok, T1} = Tests1,
Opts1= rebar_prv_common_test:translate_paths(NewState1, T1),
- %% check thath extra src dir is added
+ %% check that extra src dir is added
[App1] = rebar_state:project_apps(NewState1),
["test"] = rebar_dir:extra_src_dirs(rebar_app_info:opts(App1)),
@@ -1343,9 +1347,9 @@ testspec(Config) ->
{ok, NewState2} = rebar_prv_common_test:compile(State2, Tests2),
Opts2= rebar_prv_common_test:translate_paths(NewState2, T2),
- %% check thath extra src dirs are added
+ %% check that extra src dirs are added
[App2] = rebar_state:project_apps(NewState2),
- ["specs","test"] =
+ ["specs","suites","test"] =
lists:sort(rebar_dir:extra_src_dirs(rebar_app_info:opts(App2))),
%% check that paths are translated
@@ -1357,6 +1361,97 @@ testspec(Config) ->
ok.
+testspec_at_root(Config) ->
+ C = rebar_test_utils:init_rebar_state(Config, "ct_testspec_at_root_"),
+
+ AppDir = ?config(apps, C),
+
+ Name = rebar_test_utils:create_random_name("ct_testspec_at_root_"),
+ Vsn = rebar_test_utils:create_random_vsn(),
+ AppDir1 = filename:join([AppDir, "apps", Name]),
+ rebar_test_utils:create_app(AppDir1, Name, Vsn, [kernel, stdlib]),
+
+ Spec1 = filename:join([AppDir, "root.spec"]),
+ ok = filelib:ensure_dir(Spec1),
+ ok = file:write_file(Spec1, "{suites,\"test\",all}."),
+ Spec2 = filename:join([AppDir, "root1.spec"]),
+ ok = file:write_file(Spec2, "{suites,\".\",all}."),
+ Spec3 = filename:join([AppDir, "root2.spec"]),
+ ok = file:write_file(Spec3, "{suites,\"suites\",all}."),
+ Suite1 = filename:join(AppDir,"root_SUITE.erl"),
+ ok = file:write_file(Suite1, test_suite("root")),
+ Suite2 = filename:join([AppDir,"suites","test_SUITE.erl"]),
+ ok = filelib:ensure_dir(Suite2),
+ ok = file:write_file(Suite2, test_suite("test")),
+
+ {ok, State} = rebar_test_utils:run_and_check(C,
+ [],
+ ["as", "test", "lock"],
+ return),
+
+ Providers = rebar_state:providers(State),
+ Namespace = rebar_state:namespace(State),
+ CommandProvider = providers:get_provider(ct, Providers, Namespace),
+ GetOptSpec = providers:opts(CommandProvider),
+
+ SpecArg1 = string:join([Spec1,Spec2,Spec3],","),
+ {ok, GetOptResult1} = getopt:parse(GetOptSpec, ["--spec",SpecArg1]),
+ State1 = rebar_state:command_parsed_args(State, GetOptResult1),
+ Tests1 = rebar_prv_common_test:prepare_tests(State1),
+ {ok, NewState1} = rebar_prv_common_test:compile(State1, Tests1),
+ {ok, T1} = Tests1,
+ Opts1= rebar_prv_common_test:translate_paths(NewState1, T1),
+
+ %% check that extra src dir is added
+ ExtraDir = filename:join([AppDir, "_build", "test", "extras"]),
+ [ExtraDir,"suites","test"] =
+ rebar_dir:extra_src_dirs(rebar_state:opts(NewState1)),
+
+ %% check that path is translated
+ ExpectedSpec1 = filename:join([AppDir, "_build", "test",
+ "extras", "root.spec"]),
+ ExpectedSpec2 = filename:join([AppDir, "_build", "test",
+ "extras", "root1.spec"]),
+ ExpectedSpec3 = filename:join([AppDir, "_build", "test",
+ "extras", "root2.spec"]),
+ [ExpectedSpec1,ExpectedSpec2,ExpectedSpec3] =
+ lists:sort(proplists:get_value(spec, Opts1)),
+
+ %% check that test specs are copied
+ [ExpectedSpec1,ExpectedSpec2,ExpectedSpec3] =
+ lists:sort(filelib:wildcard(filename:join([AppDir, "_build", "test",
+ "extras", "*.spec"]))),
+
+ %% Same test again, using relative path
+ {ok,Cwd} = file:get_cwd(),
+ ok = file:set_cwd(AppDir),
+ ok = rebar_file_utils:rm_rf("_build"),
+
+ SpecArg2 = "root.spec,root1.spec,root2.spec",
+ {ok, GetOptResult2} = getopt:parse(GetOptSpec, ["--spec",SpecArg2]),
+ State2 = rebar_state:command_parsed_args(State, GetOptResult2),
+ Tests2 = rebar_prv_common_test:prepare_tests(State2),
+ {ok, NewState2} = rebar_prv_common_test:compile(State2, Tests2),
+ {ok, T2} = Tests2,
+ Opts2= rebar_prv_common_test:translate_paths(NewState2, T2),
+
+ %% check that extra src dir is added
+ [ExtraDir,"suites","test"] =
+ rebar_dir:extra_src_dirs(rebar_state:opts(NewState2)),
+
+ %% check that path is translated
+ [ExpectedSpec1,ExpectedSpec2,ExpectedSpec3] =
+ lists:sort(proplists:get_value(spec, Opts2)),
+
+ %% check that test specs are copied
+ [ExpectedSpec1,ExpectedSpec2,ExpectedSpec3] =
+ lists:sort(filelib:wildcard(filename:join([AppDir, "_build", "test",
+ "extras", "root*.spec"]))),
+
+ ok = file:set_cwd(Cwd),
+
+ ok.
+
cmd_vs_cfg_opts(Config) ->
C = rebar_test_utils:init_rebar_state(Config, "ct_cmd_vs_cfg_opts_"),