diff options
| author | Peter Andersson <[email protected]> | 2011-09-28 17:43:26 +0200 | 
|---|---|---|
| committer | Peter Andersson <[email protected]> | 2011-09-28 22:53:44 +0200 | 
| commit | b7f1a0861daa57db115b358927293d98e1630810 (patch) | |
| tree | f873f49ddeda26ff656bc7e2394028daee118684 | |
| parent | 001f0a4baa95ccc6eb553cad9e4551b4e520dc8b (diff) | |
| download | otp-b7f1a0861daa57db115b358927293d98e1630810.tar.gz otp-b7f1a0861daa57db115b358927293d98e1630810.tar.bz2 otp-b7f1a0861daa57db115b358927293d98e1630810.zip | |
Add missing tests for timetrap handling and fix remaining errors
OTP-9593
9 files changed, 888 insertions, 24 deletions
| diff --git a/lib/common_test/src/ct_framework.erl b/lib/common_test/src/ct_framework.erl index 37ecf4f928..482c5242ce 100644 --- a/lib/common_test/src/ct_framework.erl +++ b/lib/common_test/src/ct_framework.erl @@ -204,12 +204,14 @@ init_tc2(Mod,Func,SuiteInfo,MergeResult,Config,DoInit) ->  			   node=node(),  			   data={Mod,FuncSpec}}), -    case configure(MergedInfo1,MergedInfo1,SuiteInfo,{Func,DoInit},Config) of +    case catch configure(MergedInfo1,MergedInfo1,SuiteInfo,{Func,DoInit},Config) of  	{suite0_failed,Reason} ->  	    ct_util:set_testdata({curr_tc,{Mod,{suite0_failed,{require,Reason}}}}),  	    {skip,{require_failed_in_suite0,Reason}};  	{error,Reason} ->  	    {auto_skip,{require_failed,Reason}}; +	{'EXIT',Reason} -> +	    {auto_skip,Reason};  	{ok, FinalConfig} ->  	    case MergeResult of  		{error,Reason} -> @@ -1306,6 +1308,10 @@ report(What,Data) ->  		    add_to_stats(auto_skipped);  		{_,{skipped,{require_failed,_}}} ->  		    add_to_stats(auto_skipped); +		{_,{skipped,{timetrap_error,_}}} -> +		    add_to_stats(auto_skipped); +		{_,{skipped,{invalid_time_format,_}}} -> +		    add_to_stats(auto_skipped);  		{_,{skipped,_}} ->  		    add_to_stats(user_skipped);  		{_,{SkipOrFail,_Reason}} -> diff --git a/lib/common_test/test/ct_error_SUITE.erl b/lib/common_test/test/ct_error_SUITE.erl index ebbf6f3bf3..d6ee8eed10 100644 --- a/lib/common_test/test/ct_error_SUITE.erl +++ b/lib/common_test/test/ct_error_SUITE.erl @@ -60,7 +60,8 @@ suite() -> [{ct_hooks,[ts_install_cth]}].  all() ->       [cfg_error, lib_error, no_compile, timetrap_end_conf, -     timetrap_normal, timetrap_extended]. +     timetrap_normal, timetrap_extended, timetrap_parallel, +     timetrap_fun].  groups() ->       []. @@ -228,6 +229,28 @@ timetrap_parallel(Config) when is_list(Config) ->      ok = ct_test_support:verify_events(TestEvents, Events, Config).  %%%----------------------------------------------------------------- +%%% +timetrap_fun(Config) when is_list(Config) -> +    DataDir = ?config(data_dir, Config), +    Join = fun(D, S) -> filename:join(D, "error/test/"++S) end, +    Suites = [Join(DataDir, "timetrap_4_SUITE"), +	      Join(DataDir, "timetrap_5_SUITE"), +	      Join(DataDir, "timetrap_6_SUITE"), +	      Join(DataDir, "timetrap_7_SUITE")], +    {Opts,ERPid} = setup([{suite,Suites}], Config), +    ok = ct_test_support:run(Opts, Config), +    Events = ct_test_support:get_events(ERPid, Config), + +    ct_test_support:log_events(timetrap_fun, +			       reformat(Events, ?eh), +			       ?config(priv_dir, Config), +			       Opts), + +    TestEvents = events_to_check(timetrap_fun), +    ok = ct_test_support:verify_events(TestEvents, Events, Config). + + +%%%-----------------------------------------------------------------  %%% HELP FUNCTIONS  %%%----------------------------------------------------------------- @@ -814,7 +837,7 @@ test_events(timetrap_parallel) ->      [       {?eh,start_logging,{'DEF','RUNDIR'}},       {?eh,test_start,{'DEF',{'START_TIME','LOGDIR'}}}, -     {?eh,start_info,{1,1,7}}, +     {?eh,start_info,{1,1,8}},       {?eh,tc_done,{timetrap_3_SUITE,init_per_suite,ok}},       {parallel,        [{?eh,tc_start, @@ -826,9 +849,12 @@ test_events(timetrap_parallel) ->         {?eh,tc_start,{timetrap_3_SUITE,tc2}},         {?eh,tc_start,{timetrap_3_SUITE,tc3}},         {?eh,tc_start,{timetrap_3_SUITE,tc4}}, +       {?eh,tc_start,{timetrap_3_SUITE,tc5}},         {?eh,tc_start,{timetrap_3_SUITE,tc6}},         {?eh,tc_start,{timetrap_3_SUITE,tc7}},         {?eh,tc_done, +        {timetrap_3_SUITE,tc5,ok}}, +       {?eh,tc_done,          {timetrap_3_SUITE,tc1,{failed,{timetrap_timeout,500}}}},         {?eh,tc_done,          {timetrap_3_SUITE,tc2,{failed,{timetrap_timeout,1000}}}}, @@ -842,11 +868,90 @@ test_events(timetrap_parallel) ->          {timetrap_3_SUITE,tc4,{failed,{timetrap_timeout,2000}}}},         {?eh,tc_done,          {timetrap_3_SUITE,tc3,{failed,{timetrap_timeout,3000}}}}, -       {?eh,test_stats,{0,7,{0,0}}}, +       {?eh,test_stats,{1,7,{0,0}}},         {?eh,tc_start,  	{timetrap_3_SUITE,{end_per_group,g1,[parallel]}}},         {?eh,tc_done,  	{timetrap_3_SUITE,{end_per_group,g1,[parallel]},ok}}]},       {?eh,tc_done,{timetrap_3_SUITE,end_per_suite,ok}},       {?eh,test_done,{'DEF','STOP_TIME'}}, -     {?eh,stop_logging,[]}]. +     {?eh,stop_logging,[]}]; + +test_events(timetrap_fun) -> +    [ +     {?eh,start_logging,{'DEF','RUNDIR'}}, +     {?eh,start_info,{4,4,17}}, +     {?eh,tc_done,{timetrap_4_SUITE,init_per_suite,ok}}, +     {?eh,tc_start,{timetrap_4_SUITE,tc0}}, +     {?eh,tc_done, +      {timetrap_4_SUITE,tc0,{failed,{timetrap_timeout,1000}}}}, +     {?eh,tc_start,{timetrap_4_SUITE,tc1}}, +     {?eh,tc_done, +      {timetrap_4_SUITE,tc1,{failed,{timetrap_timeout,2000}}}}, +     {?eh,tc_start,{timetrap_4_SUITE,tc2}}, +     {?eh,tc_done, +      {timetrap_4_SUITE,tc2,{failed,{timetrap_timeout,500}}}}, +     {?eh,tc_start,{timetrap_4_SUITE,tc3}}, +     {?eh,tc_done, +      {timetrap_4_SUITE,tc3,{failed,{timetrap_timeout,1000}}}}, +     {?eh,test_stats,{0,4,{0,0}}}, +     {?eh,tc_done,{timetrap_4_SUITE,end_per_suite,ok}}, + +     {?eh,tc_done,{timetrap_5_SUITE,init_per_suite,ok}}, +     {?eh,tc_start,{timetrap_5_SUITE,tc0}}, +     {?eh,tc_done, +      {timetrap_5_SUITE,tc0,{failed,{timetrap_timeout,1000}}}}, +     {?eh,test_stats,{0,5,{0,0}}}, +     {?eh,tc_start,{timetrap_5_SUITE,tc1}}, +     {?eh,tc_done, +      {timetrap_5_SUITE,tc1,{skipped,{timetrap_error,kaboom}}}}, +     {?eh,tc_start,{timetrap_5_SUITE,tc2}}, +     {?eh,tc_done, +      {timetrap_5_SUITE,tc2,{skipped,{timetrap_error,kaboom}}}}, +     {?eh,tc_start,{timetrap_5_SUITE,tc3}}, +     {?eh,tc_done, +      {timetrap_5_SUITE,tc3, +       {skipped,{invalid_time_format,{timetrap_utils,timetrap_val,[5000]}}}}}, +     {?eh,tc_start,{timetrap_5_SUITE,tc4}}, +     {?eh,tc_done, +      {timetrap_5_SUITE,tc4,{skipped,{invalid_time_format,'_'}}}}, +     {?eh,test_stats,{0,5,{0,4}}}, +     {?eh,tc_start,{timetrap_5_SUITE,tc5}}, +     {?eh,tc_done, +      {timetrap_5_SUITE,tc5,{failed,{timetrap_timeout,1000}}}}, +     {?eh,tc_start,{timetrap_5_SUITE,tc6}}, +     {?eh,tc_done, +      {timetrap_5_SUITE,tc6,{failed,{timetrap_timeout,1000}}}}, +     {?eh,tc_start,{timetrap_5_SUITE,tc7}}, +     {?eh,tc_done, +      {timetrap_5_SUITE,tc7,{failed,{timetrap_timeout,1000}}}}, +     {?eh,test_stats,{0,8,{0,4}}}, +     {?eh,tc_done,{timetrap_5_SUITE,end_per_suite,ok}}, + +     {?eh,tc_start,{timetrap_6_SUITE,init_per_suite}}, +     {?eh,tc_done, +      {timetrap_6_SUITE,init_per_suite,{skipped,{timetrap_error,kaboom}}}}, +     {?eh,tc_auto_skip, +      {timetrap_6_SUITE,tc0,{fw_auto_skip,{timetrap_error,kaboom}}}}, +     {?eh,test_stats,{0,8,{0,5}}}, +     {?eh,tc_auto_skip, +      {timetrap_6_SUITE,end_per_suite,{fw_auto_skip,{timetrap_error,kaboom}}}}, + +     {?eh,tc_done,{timetrap_7_SUITE,init_per_suite,ok}}, +     {?eh,tc_start,{timetrap_7_SUITE,tc0}}, +     {?eh,tc_done, +      {timetrap_7_SUITE,tc0,{failed,{timetrap_timeout,1000}}}}, +     {?eh,tc_start,{timetrap_7_SUITE,tc1}}, +     {?eh,tc_done, +      {timetrap_7_SUITE,tc1,{failed,{timetrap_timeout,2000}}}}, +     {?eh,tc_start,{timetrap_7_SUITE,tc2}}, +     {?eh,tc_done, +      {timetrap_7_SUITE,tc2,{failed,{timetrap_timeout,500}}}}, +     {?eh,tc_start,{timetrap_7_SUITE,tc3}}, +     {?eh,tc_done, +      {timetrap_7_SUITE,tc3,{failed,{timetrap_timeout,1000}}}}, +     {?eh,test_stats,{0,12,{0,5}}}, +     {?eh,tc_done,{timetrap_7_SUITE,end_per_suite,ok}}, +     {?eh,test_done,{'DEF','STOP_TIME'}}, +     {?eh,stop_logging,[]} +    ]. diff --git a/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_3_SUITE.erl b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_3_SUITE.erl new file mode 100644 index 0000000000..8271b23afe --- /dev/null +++ b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_3_SUITE.erl @@ -0,0 +1,146 @@ +%% +%% %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% +%% +-module(timetrap_3_SUITE). + +-compile(export_all). + +-include_lib("common_test/include/ct.hrl"). + +-define(TO, 3). + +%%-------------------------------------------------------------------- +%% Function: suite() -> Info +%% Info = [tuple()] +%%-------------------------------------------------------------------- +suite() -> +    [{timetrap,{seconds,?TO}}]. + +%%-------------------------------------------------------------------- +%% Function: init_per_suite(Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_suite(Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_suite(Config0) -> void() | {save_config,Config1} +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_suite(_Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: init_per_group(GroupName, Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% GroupName = atom() +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_group(_GroupName, Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_group(GroupName, Config0) -> +%%               void() | {save_config,Config1} +%% GroupName = atom() +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_group(_GroupName, _Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: groups() -> [Group] +%% Group = {GroupName,Properties,GroupsAndTestCases} +%% GroupName = atom() +%% Properties = [parallel | sequence | Shuffle | {RepeatType,N}] +%% GroupsAndTestCases = [Group | {group,GroupName} | TestCase] +%% TestCase = atom() +%% Shuffle = shuffle | {shuffle,{integer(),integer(),integer()}} +%% RepeatType = repeat | repeat_until_all_ok | repeat_until_all_fail | +%%              repeat_until_any_ok | repeat_until_any_fail +%% N = integer() | forever +%%-------------------------------------------------------------------- +groups() -> +    [{g1,[parallel],[tc0,tc1,tc2,tc3,tc4,tc5,tc6,tc7]}]. + +%%-------------------------------------------------------------------- +%% Function: all() -> GroupsAndTestCases | {skip,Reason} +%% GroupsAndTestCases = [{group,GroupName} | TestCase] +%% GroupName = atom() +%% TestCase = atom() +%% Reason = term() +%%-------------------------------------------------------------------- +all() -> +    [{group,g1}]. + +tc0() -> +    [{timetrap,2000}]. +tc0(_) -> +    ct:comment("TO after 2 sec"), +    ct:sleep({seconds,5}), +    ok. + +tc1() -> +    [{timetrap,500}]. +tc1(_) -> +    ct:comment("TO after 1/2 sec"), +    ct:sleep({seconds,5}), +    ok. + +tc2() -> +    [{timetrap,1000}]. +tc2(_) -> +    ct:comment("TO after 1 sec"), +    ct:sleep({seconds,5}), +    ok. + +tc3(_) ->     +    ct:comment(io_lib:format("TO after ~w sec", [?TO])), +    ct:sleep({seconds,5}), +    ok. + +tc4() -> +    [{timetrap,2000}]. +tc4(_) -> +    ct:comment(io_lib:format("TO after 2 sec", [])), +    ct:sleep({seconds,5}), +    ok. + +tc5() -> +    [{timetrap,2000}]. +tc5(_) -> +    ct:comment("No timeout"), +    ct:sleep({seconds,1}), +    ok. + +tc6() -> +    [{timetrap,1000}]. +tc6(_) -> +    ct:comment("TO after 1 sec"), +    ct:sleep({seconds,5}), +    ok. + +tc7() -> +    [{timetrap,1500}]. +tc7(_) -> +    ct:comment("TO after 1 1/2 sec"), +    ct:sleep({seconds,5}), +    ok. diff --git a/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_4_SUITE.erl b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_4_SUITE.erl new file mode 100644 index 0000000000..d902454f09 --- /dev/null +++ b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_4_SUITE.erl @@ -0,0 +1,135 @@ +%% +%% %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% +%% +-module(timetrap_4_SUITE). + +-compile(export_all). + +-include_lib("common_test/include/ct.hrl"). + +-define(TO, 1). + +%%-------------------------------------------------------------------- +%% Function: suite() -> Info +%% Info = [tuple()] +%%-------------------------------------------------------------------- +suite() -> +    [{timetrap,{timetrap_utils,timetrap_val,[{seconds,?TO}]}}]. + +%%-------------------------------------------------------------------- +%% Function: init_per_suite(Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_suite(Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_suite(Config0) -> void() | {save_config,Config1} +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_suite(_Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: init_per_group(GroupName, Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% GroupName = atom() +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_group(_GroupName, Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_group(GroupName, Config0) -> +%%               void() | {save_config,Config1} +%% GroupName = atom() +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_group(_GroupName, _Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: init_per_testcase(TestCase, Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% TestCase = atom() +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_testcase(_, Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_testcase(TestCase, Config0) -> +%%               void() | {save_config,Config1} +%% TestCase = atom() +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_testcase(_, _Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: groups() -> [Group] +%% Group = {GroupName,Properties,GroupsAndTestCases} +%% GroupName = atom() +%% Properties = [parallel | sequence | Shuffle | {RepeatType,N}] +%% GroupsAndTestCases = [Group | {group,GroupName} | TestCase] +%% TestCase = atom() +%% Shuffle = shuffle | {shuffle,{integer(),integer(),integer()}} +%% RepeatType = repeat | repeat_until_all_ok | repeat_until_all_fail | +%%              repeat_until_any_ok | repeat_until_any_fail +%% N = integer() | forever +%%-------------------------------------------------------------------- +groups() -> +    []. + +%%-------------------------------------------------------------------- +%% Function: all() -> GroupsAndTestCases | {skip,Reason} +%% GroupsAndTestCases = [{group,GroupName} | TestCase] +%% GroupName = atom() +%% TestCase = atom() +%% Reason = term() +%%-------------------------------------------------------------------- +all() -> +    [tc0,tc1,tc2,tc3]. + +tc0(_) -> +    ct:comment(io_lib:format("TO after ~w sec", [?TO])), +    ct:sleep({seconds,5}), +    ok. + +tc1() -> +    [{timetrap,{timetrap_utils,timetrap_val,[2000]}}]. +tc1(_) -> +    ct:comment("TO after 2 sec"), +    ct:sleep({seconds,5}), +    ok. + +tc2() -> +    [{timetrap,fun() -> timetrap_utils:timetrap_val(500) end}]. +tc2(_) -> +    ct:comment("TO after 0.5 sec"), +    ct:sleep(1000), +    ok. + +tc3(_) -> +    ct:comment(io_lib:format("TO after ~w sec", [?TO])), +    ct:sleep({seconds,5}), +    ok. diff --git a/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_5_SUITE.erl b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_5_SUITE.erl new file mode 100644 index 0000000000..c5d4b5062e --- /dev/null +++ b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_5_SUITE.erl @@ -0,0 +1,155 @@ +%% +%% %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% +%% +-module(timetrap_5_SUITE). + +-compile(export_all). + +-include_lib("common_test/include/ct.hrl"). + +-define(TO, 1). + +%%-------------------------------------------------------------------- +%% Function: suite() -> Info +%% Info = [tuple()] +%%-------------------------------------------------------------------- +suite() -> +    [{timetrap, fun() -> timetrap_utils:timetrap_val({seconds,?TO}) end}]. + +%%-------------------------------------------------------------------- +%% Function: init_per_suite(Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_suite(Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_suite(Config0) -> void() | {save_config,Config1} +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_suite(_Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: init_per_group(GroupName, Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% GroupName = atom() +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_group(_GroupName, Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_group(GroupName, Config0) -> +%%               void() | {save_config,Config1} +%% GroupName = atom() +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_group(_GroupName, _Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: init_per_testcase(TestCase, Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% TestCase = atom() +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_testcase(_, Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_testcase(TestCase, Config0) -> +%%               void() | {save_config,Config1} +%% TestCase = atom() +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_testcase(_, _Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: groups() -> [Group] +%% Group = {GroupName,Properties,GroupsAndTestCases} +%% GroupName = atom() +%% Properties = [parallel | sequence | Shuffle | {RepeatType,N}] +%% GroupsAndTestCases = [Group | {group,GroupName} | TestCase] +%% TestCase = atom() +%% Shuffle = shuffle | {shuffle,{integer(),integer(),integer()}} +%% RepeatType = repeat | repeat_until_all_ok | repeat_until_all_fail | +%%              repeat_until_any_ok | repeat_until_any_fail +%% N = integer() | forever +%%-------------------------------------------------------------------- +groups() -> +    []. + +%%-------------------------------------------------------------------- +%% Function: all() -> GroupsAndTestCases | {skip,Reason} +%% GroupsAndTestCases = [{group,GroupName} | TestCase] +%% GroupName = atom() +%% TestCase = atom() +%% Reason = term() +%%-------------------------------------------------------------------- +all() -> +    [tc0,tc1,tc2,tc3,tc4,tc5,tc6,tc7]. + +tc0(_) -> +    ct:comment(io_lib:format("TO after ~w sec", [?TO])), +    ct:sleep({seconds,5}), +    ok. + +tc1() -> +    [{timetrap,{timetrap_utils,timetrap_exit,[kaboom]}}]. +tc1(_) -> +    exit(this_should_not_execute). + +tc2() -> +    [{timetrap,fun() -> exit(kaboom) end}]. +tc2(_) -> +    exit(this_should_not_execute). + +tc3() -> +    [{timetrap,{timetrap_utils,timetrap_err_mfa,[]}}]. +tc3(_) -> +    exit(this_should_not_execute). + +tc4() -> +    [{timetrap,fun() -> timetrap_utils:timetrap_err_fun() end}]. +tc4(_) -> +    exit(this_should_not_execute). + +tc5() -> +    [{timetrap,{timetrap_utils,timetrap_timeout,[{seconds,40}, +						 {seconds,1}]}}]. +tc5(_) ->     +    ct:comment("TO after 40+1 sec"), +    ct:sleep({seconds,42}), +    ok. + +tc6() -> +    [{timetrap,fun() -> ct:sleep(6000), 1000 end}]. +tc6(_) -> +    ct:comment("TO after 6+1 sec"), +    ct:sleep({seconds,10}). + +tc7(_) -> +    ct:comment(io_lib:format("TO after ~w sec", [?TO])), +    ct:sleep({seconds,5}), +    ok. diff --git a/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_6_SUITE.erl b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_6_SUITE.erl new file mode 100644 index 0000000000..90467ff752 --- /dev/null +++ b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_6_SUITE.erl @@ -0,0 +1,114 @@ +%% +%% %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% +%% +-module(timetrap_6_SUITE). + +-compile(export_all). + +-include_lib("common_test/include/ct.hrl"). + +-define(TO, 1). + +%%-------------------------------------------------------------------- +%% Function: suite() -> Info +%% Info = [tuple()] +%%-------------------------------------------------------------------- +suite() -> +    [{timetrap, fun() -> exit(kaboom) end}]. + +%%-------------------------------------------------------------------- +%% Function: init_per_suite(Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_suite(Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_suite(Config0) -> void() | {save_config,Config1} +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_suite(_Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: init_per_group(GroupName, Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% GroupName = atom() +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_group(_GroupName, Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_group(GroupName, Config0) -> +%%               void() | {save_config,Config1} +%% GroupName = atom() +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_group(_GroupName, _Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: init_per_testcase(TestCase, Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% TestCase = atom() +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_testcase(_, Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_testcase(TestCase, Config0) -> +%%               void() | {save_config,Config1} +%% TestCase = atom() +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_testcase(_, _Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: groups() -> [Group] +%% Group = {GroupName,Properties,GroupsAndTestCases} +%% GroupName = atom() +%% Properties = [parallel | sequence | Shuffle | {RepeatType,N}] +%% GroupsAndTestCases = [Group | {group,GroupName} | TestCase] +%% TestCase = atom() +%% Shuffle = shuffle | {shuffle,{integer(),integer(),integer()}} +%% RepeatType = repeat | repeat_until_all_ok | repeat_until_all_fail | +%%              repeat_until_any_ok | repeat_until_any_fail +%% N = integer() | forever +%%-------------------------------------------------------------------- +groups() -> +    []. + +%%-------------------------------------------------------------------- +%% Function: all() -> GroupsAndTestCases | {skip,Reason} +%% GroupsAndTestCases = [{group,GroupName} | TestCase] +%% GroupName = atom() +%% TestCase = atom() +%% Reason = term() +%%-------------------------------------------------------------------- +all() -> +    [tc0]. + +tc0(_) -> +    ok. diff --git a/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_7_SUITE.erl b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_7_SUITE.erl new file mode 100644 index 0000000000..b25b7770a7 --- /dev/null +++ b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_7_SUITE.erl @@ -0,0 +1,137 @@ +%% +%% %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% +%% +-module(timetrap_7_SUITE). + +-compile(export_all). + +-include_lib("common_test/include/ct.hrl"). + +-define(TO, 1). +-define(HANG, 6). + +%%-------------------------------------------------------------------- +%% Function: suite() -> Info +%% Info = [tuple()] +%%-------------------------------------------------------------------- +suite() -> +    [{timetrap,{timetrap_utils,timetrap_timeout,[{seconds,?HANG}, +						 {seconds,?TO}]}}]. + +%%-------------------------------------------------------------------- +%% Function: init_per_suite(Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_suite(Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_suite(Config0) -> void() | {save_config,Config1} +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_suite(_Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: init_per_group(GroupName, Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% GroupName = atom() +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_group(_GroupName, Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_group(GroupName, Config0) -> +%%               void() | {save_config,Config1} +%% GroupName = atom() +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_group(_GroupName, _Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: init_per_testcase(TestCase, Config0) -> +%%               Config1 | {skip,Reason} | {skip_and_save,Reason,Config1} +%% TestCase = atom() +%% Config0 = Config1 = [tuple()] +%% Reason = term() +%%-------------------------------------------------------------------- +init_per_testcase(_, Config) -> +    Config. + +%%-------------------------------------------------------------------- +%% Function: end_per_testcase(TestCase, Config0) -> +%%               void() | {save_config,Config1} +%% TestCase = atom() +%% Config0 = Config1 = [tuple()] +%%-------------------------------------------------------------------- +end_per_testcase(_, _Config) -> +    ok. + +%%-------------------------------------------------------------------- +%% Function: groups() -> [Group] +%% Group = {GroupName,Properties,GroupsAndTestCases} +%% GroupName = atom() +%% Properties = [parallel | sequence | Shuffle | {RepeatType,N}] +%% GroupsAndTestCases = [Group | {group,GroupName} | TestCase] +%% TestCase = atom() +%% Shuffle = shuffle | {shuffle,{integer(),integer(),integer()}} +%% RepeatType = repeat | repeat_until_all_ok | repeat_until_all_fail | +%%              repeat_until_any_ok | repeat_until_any_fail +%% N = integer() | forever +%%-------------------------------------------------------------------- +groups() -> +    []. + +%%-------------------------------------------------------------------- +%% Function: all() -> GroupsAndTestCases | {skip,Reason} +%% GroupsAndTestCases = [{group,GroupName} | TestCase] +%% GroupName = atom() +%% TestCase = atom() +%% Reason = term() +%%-------------------------------------------------------------------- +all() -> +    [tc0,tc1,tc2,tc3]. + +tc0(_) -> +    ct:comment(io_lib:format("TO after ~w+~w sec", [?HANG,?TO])), +    ct:sleep({seconds,5}), +    ok. + +tc1() -> +    [{timetrap,{timetrap_utils,timetrap_val,[2000]}}]. +tc1(_) -> +    ct:comment("TO after 2 sec"), +    ct:sleep({seconds,5}), +    ok. + +tc2() -> +    [{timetrap,fun() -> timetrap_utils:timetrap_val(500) end}]. +tc2(_) -> +    ct:comment("TO after 0.5 sec"), +    ct:sleep(1000), +    ok. + +tc3(_) -> +    ct:comment(io_lib:format("TO after ~w+~w sec", [?HANG,?TO])), +    ct:sleep({seconds,5}), +    ok. diff --git a/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_utils.erl b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_utils.erl new file mode 100644 index 0000000000..fcde6cd701 --- /dev/null +++ b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_utils.erl @@ -0,0 +1,43 @@ +%% +%% %CopyrightBegin% +%% +%% Copyright Ericsson AB 2009-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% +%% + +-module(timetrap_utils). + +-export([timetrap_val/1, +	 timetrap_err_fun/0, +	 timetrap_err_mfa/0, +	 timetrap_exit/1, +	 timetrap_timeout/2]). + +timetrap_val(Val) -> +    Val. + +timetrap_err_fun() -> +    fun() -> 5000 end. + +timetrap_err_mfa() -> +    {?MODULE,timetrap_val,[5000]}. + +timetrap_exit(Reason) -> +    exit(Reason). + +timetrap_timeout(Sleep, Val) -> +    ct:sleep(Sleep), +    Val. +	     diff --git a/lib/test_server/src/test_server.erl b/lib/test_server/src/test_server.erl index e1a139af6f..12625e3c00 100644 --- a/lib/test_server/src/test_server.erl +++ b/lib/test_server/src/test_server.erl @@ -1921,30 +1921,53 @@ time_ms({seconds,N}) -> seconds(N);  time_ms({Other,_N}) ->      format("=== ERROR: Invalid time specification: ~p. "  	   "Should be seconds, minutes, or hours.~n", [Other]), -    exit({invalid_time_spec,Other}); +    exit({invalid_time_format,Other});  time_ms(Ms) when is_integer(Ms) -> Ms;  time_ms(infinity) -> infinity;  time_ms(Fun) when is_function(Fun) -> -    try Fun() of -	Val -> time_ms1(Val) -    catch -	_:Error -> -	    exit({timetrap_error,Error}) -    end; -time_ms({M,F,A}) when is_atom(M), is_atom(F), is_list(A) -> -    try apply(M, F, A) of -	Val -> time_ms1(Val) -    catch -	_:Error -> +    time_ms_apply(Fun); +time_ms({M,F,A}=MFA) when is_atom(M), is_atom(F), is_list(A) -> +    time_ms_apply(MFA); +time_ms(Other) -> exit({invalid_time_format,Other}). + +time_ms_apply(Func) -> +    time_ms_apply(Func, [5000,30000,60000,infinity]). + +time_ms_apply(Func, TOs) -> +    Apply = fun() -> +		    case Func of +			{M,F,A} -> +			    exit({self(),apply(M, F, A)}); +			Fun -> +			    exit({self(),Fun()}) +		    end +	    end, +    Pid = spawn(Apply), +    Ref = monitor(process, Pid), +    time_ms_wait(Func, Pid, Ref, TOs). + +time_ms_wait(Func, Pid, Ref, [TO|TOs]) -> +    receive +	{'DOWN',Ref,process,Pid,{Pid,Result}} -> +	    time_ms_check(Result); +	{'DOWN',Ref,process,Pid,Error} ->  	    exit({timetrap_error,Error}) +    after +	TO -> +	    format("=== WARNING: No return from timetrap function ~p~n", [Func]), +	    time_ms_wait(Func, Pid, Ref, TOs)      end; -time_ms(Other) -> exit({invalid_time_spec,Other}). - -time_ms1(MFA = {M,F,A}) when is_atom(M), is_atom(F), is_list(A) -> -    exit({invalid_time_spec,MFA}); -time_ms1(Fun) when is_function(Fun) -> -    exit({invalid_time_spec,Fun}); -time_ms1(Other) -> +%% this clause will never execute if 'infinity' is in TOs list, that's ok! +time_ms_wait(Func, Pid, Ref, []) -> +    demonitor(Ref), +    exit(Pid, kill), +    exit({timetrap_error,{no_return_from_timetrap_function,Func}}). + +time_ms_check(MFA = {M,F,A}) when is_atom(M), is_atom(F), is_list(A) -> +    exit({invalid_time_format,MFA}); +time_ms_check(Fun) when is_function(Fun) -> +    exit({invalid_time_format,Fun}); +time_ms_check(Other) ->      time_ms(Other).  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% | 
