%%
%% %CopyrightBegin%
%%
%% Copyright Ericsson AB 2010-2011. All Rights Reserved.
%%
%% The contents of this file are subject to the Erlang Public License,
%% Version 1.1, (the "License"); you may not use this file except in
%% compliance with the License. You should have received a copy of the
%% Erlang Public License along with this software. If not, it can be
%% retrieved online at http://www.erlang.org/.
%%
%% 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.
%%
%% %CopyrightEnd%
%%
%%
%% Tests based on the contents of the diameter app file.
%%
-module(diameter_app_SUITE).
-export([suite/0,
all/0,
init_per_suite/1,
end_per_suite/1]).
%% testcases
-export([keys/1,
modules/1,
exports/1,
applications/1,
undefined_calls/1, undefined_calls/0]).
-define(APP, diameter).
-define(A, list_to_atom).
%% ===========================================================================
suite() ->
[{timetrap, {seconds, 10}}].
all() ->
[keys,
modules,
exports,
applications,
undefined_calls].
init_per_suite(Config) ->
{ok, App} = diameter_util:appfile(?APP),
[{app, App} | Config].
end_per_suite(_Config) ->
ok.
%% ===========================================================================
%% keys/1
%%
%% Ensure that the app file contain required keys.
keys(Config) ->
App = fetch(app, Config),
[] = lists:filter(fun(K) -> not lists:keymember(K, 1, App) end,
[vsn, description, modules, registered, applications]).
%% modules/1
%%
%% Ensure that the app file module list match the installed beams.
modules(Config) ->
Mods = fetch(modules, fetch(app, Config)),
Installed = installed_mods(),
{[], []} = {Mods -- Installed, Installed -- Mods}.
installed_mods() ->
Dir = code:lib_dir(?APP, ebin),
{ok, Files} = file:list_dir(Dir),
[?A(lists:reverse(R)) || N <- Files, "maeb." ++ R <- [lists:reverse(N)]].
%% exports/1
%%
%% Ensure that no module does export_all.
exports(Config) ->
Mods = fetch(modules, fetch(app, Config)),
[] = [M || M <- Mods, exports_all(M)].
exports_all(Mod) ->
Opts = fetch(options, Mod:module_info(compile)),
is_list(Opts) andalso lists:member(export_all, Opts).
%% applications/1
%%
%% Ensure that each dependent application is on the code path.
applications(Config) ->
As = fetch(applications, fetch(app, Config)),
[] = [A || A <- As, {error, _} <- [diameter_util:appfile(A)]].
%% undefined_calls/1
%%
%% Ensure that no function on our application calls an undefined function.
undefined_calls() ->
[{timetrap, {minutes, 2}}].
undefined_calls(Config) ->
Mods = fetch(modules, fetch(app, Config)),
RootDir = code:root_dir(),
EbinDir = code:lib_dir(?APP, ebin),
{ok, XRef} = xref:start(make_name(xref_test_name)),
ok = xref:set_default(XRef, [{verbose, false}, {warnings, false}]),
XRefName = make_name(xref_name),
{ok, XRefName} = xref:add_release(XRef, RootDir, {name, XRefName}),
{ok, _} = xref:replace_application(XRef, ?APP, EbinDir),
{ok, Undefs} = xref:analyze(XRef, undefined_function_calls),
xref:stop(XRef),
[] = lists:filter(fun({{M,_,_},_}) -> lists:member(M, Mods) end, Undefs).
make_name(Suf) ->
list_to_atom(atom_to_list(?APP) ++ "_" ++ atom_to_list(Suf)).
%% ===========================================================================
fetch(Key, List) ->
{Key, Val} = lists:keyfind(Key, 1, List),
Val.