diff options
Diffstat (limited to 'lib/common_test')
50 files changed, 3297 insertions, 688 deletions
| diff --git a/lib/common_test/doc/src/ct_master_chapter.xml b/lib/common_test/doc/src/ct_master_chapter.xml index 37a0805055..adfe79e41a 100644 --- a/lib/common_test/doc/src/ct_master_chapter.xml +++ b/lib/common_test/doc/src/ct_master_chapter.xml @@ -198,7 +198,7 @@    <section>      <title>Automatic startup of test target nodes</title>      <marker id="ct_slave"></marker> -    <p>Is is possible to automatically start, and perform initial actions, on +    <p>It is possible to automatically start, and perform initial actions, on        test target nodes by using the test specification term <c>init</c>.</p>      <p>Currently, two sub-terms are supported, <c>node_start</c> and <c>eval</c>.</p>      <p>Example:</p> diff --git a/lib/common_test/doc/src/install_chapter.xml b/lib/common_test/doc/src/install_chapter.xml index 7f8c606324..31125b945c 100644 --- a/lib/common_test/doc/src/install_chapter.xml +++ b/lib/common_test/doc/src/install_chapter.xml @@ -34,7 +34,7 @@      <title>General information</title>      <p>The two main interfaces for running tests with Common Test -      are an executable program named ct_run and an +      are an executable program named <c>ct_run</c> and an        erlang module named <c>ct</c>. The ct_run program        is compiled for the underlying operating system (e.g. Unix/Linux        or Windows) during the build of the Erlang/OTP system, and is @@ -43,67 +43,10 @@        The <c>ct</c> interface functions can be called from the Erlang shell,        or from any Erlang function, on any supported platform.</p> -    <p>A legacy Bourne shell script - named run_test - exists, -      which may be manually generated and installed. This script may be used -      instead of the ct_run program mentioned above, e.g. if the user -      wishes to modify or customize the Common Test start flags in a simpler -      way than making changes to the ct_run C program.</p> -      <p>The Common Test application is installed with the Erlang/OTP        system and no additional installation step is required to start using -      Common Test by means of the ct_run executable program, and/or the interface -      functions in the <c>ct</c> module. If you wish to use the legacy Bourne -      shell script version run_test, however, this script needs to be -      generated first, according to the instructions below.</p> - -    <note><p>Before reading on, please note that since Common Test version -	1.5, the run_test shell script is no longer required for starting -	tests with Common Test from the OS command line. The ct_run -	program (descibed above) is the new recommended command line interface -	for Common Test. The shell script exists mainly for legacy reasons and -	may not be updated in future releases of Common Test. It may even be removed. -    </p></note> - -    <p>Optional step to generate a shell script for starting Common Test:</p> -    <p>To generate the run_test shell script, navigate to the -      <c><![CDATA[common_test-<vsn>]]></c> directory, located among the other -      OTP applications (under the OTP lib directory). Here execute the -      <c>install.sh</c> script with argument <c>local</c>:</p> - -    <p><c> -	$ ./install.sh local -    </c></p> -     -    <p>This generates the executable run_test script in the -      <c><![CDATA[common_test-<vsn>/priv/bin]]></c> directory. The script -      will include absolute paths to the Common Test and Test Server -      application directories, so it's possible to copy or move the script to -      a different location on the file system, if desired, without having to -      update it. It's of course possible to leave the script under the -      <c>priv/bin</c> directory and update the PATH variable accordingly (or -      create a link or alias to it).</p> - -    <p>If you, for any reason, have copied Common Test and Test Server -      to a different location than the default OTP lib directory, you can -      generate a run_test script with a different top level directory, -      simply by specifying the directory, instead of <c>local</c>, when running -      <c>install.sh</c>. Example:</p> - -    <p><c> -	$ install.sh /usr/local/test_tools -    </c></p> - -    <p>Note that the <c><![CDATA[common_test-<vsn>]]></c> and -      <c><![CDATA[test_server-<vsn>]]></c> directories must be located under the -      same top directory. Note also that the install script does not copy files -      or update environment variables. It only generates the run_test -      script.</p> -     -    <p>Whenever you install a new version of Erlang/OTP, the run_test -      script needs to be regenerated, or updated manually with new directory names -      (new version numbers), for it to "see" the latest Common Test and Test Server -      versions.</p> - +      Common Test by means of the <c>ct_run</c> executable program, and/or +      the interface functions in the <c>ct</c> module.</p>      </section>  </chapter> diff --git a/lib/common_test/doc/src/notes.xml b/lib/common_test/doc/src/notes.xml index 822ebf146e..472e3b7833 100644 --- a/lib/common_test/doc/src/notes.xml +++ b/lib/common_test/doc/src/notes.xml @@ -32,6 +32,66 @@      <file>notes.xml</file>      </header> +<section><title>Common_Test 1.10.1</title> + +    <section><title>Fixed Bugs and Malfunctions</title> +      <list> +        <item> +          <p> +	    A fault in the Common Test logger process, that caused +	    the application to crash when running on a long name +	    node, has been corrected.</p> +          <p> +	    Own Id: OTP-12643</p> +        </item> +        <item> +          <p> +	    A 'wait_for_prompt' option in ct_telnet:expect/3 has been +	    introduced which forces the function to not return until +	    a prompt string has been received, even if other expect +	    patterns have already been found.</p> +          <p> +	    Own Id: OTP-12688 Aux Id: seq12818 </p> +        </item> +        <item> +          <p> +	    If the last expression in a test case causes a timetrap +	    timeout, the stack trace is ignored and not printed to +	    the test case log file. This happens because the +	    {Suite,TestCase,Line} info is not available in the stack +	    trace in this scenario, due to tail call elimination. +	    Common Test has been modified to handle this situation by +	    inserting a {Suite,TestCase,last_expr} tuple in the +	    correct place and printing the stack trace as expected.</p> +          <p> +	    Own Id: OTP-12697 Aux Id: seq12848 </p> +        </item> +        <item> +          <p> +	    Fixed a buffer problem in ct_netconfc which could cause +	    that some messages where buffered forever.</p> +          <p> +	    Own Id: OTP-12698 Aux Id: seq12844 </p> +        </item> +        <item> +          <p> +	    The VTS mode in Common Test has been modified to use a +	    private version of the Webtool application (ct_webtool).</p> +          <p> +	    Own Id: OTP-12704 Aux Id: OTP-10922 </p> +        </item> +        <item> +          <p> +	    Add possibility to add user capabilities in +	    <c>ct_netconfc:hello/3</c>.</p> +          <p> +	    Own Id: OTP-12707 Aux Id: seq12846 </p> +        </item> +      </list> +    </section> + +</section> +  <section><title>Common_Test 1.10</title>      <section><title>Fixed Bugs and Malfunctions</title> diff --git a/lib/common_test/doc/src/run_test_chapter.xml b/lib/common_test/doc/src/run_test_chapter.xml index 864f82cb63..df60e5f7f2 100644 --- a/lib/common_test/doc/src/run_test_chapter.xml +++ b/lib/common_test/doc/src/run_test_chapter.xml @@ -1005,6 +1005,31 @@  	  for starting the tests, the relaxed scanner  	  mode is enabled by means of the tuple: <c>{allow_user_terms,true}</c></p>  	</section> +	<section> +	  <title>Reading test specification terms</title> +	  <p>It's possible to look up terms in the current test specification +	  (i.e. the spec that's been used to configure and run the current test). +	  The function <c>get_testspec_terms()</c> returns a list of all test spec +	  terms (both config- and test terms) and <c>get_testspec_terms(Tags)</c> +	  returns the term (or a list of terms) matching the tag (or tags) in +	  <c>Tags</c>.</p> +	  <p>For example, in the test specification:</p> +	  <pre> +	    ... +	    {label, my_server_smoke_test}. +	    {config, "../../my_server_setup.cfg"}. +	    {config, "../../my_server_interface.cfg"}. +	    ...</pre>	     +	  <p>And in e.g. a test suite or a CT hook function:</p> +	  <pre> +	    ... +	    [{label,[{_Node,TestType}]}, {config,CfgFiles}] = +                ct:get_testspec_terms([label,config]), + +            [verify_my_server_cfg(TestType, CfgFile) || {Node,CfgFile} <- CfgFiles, +                                                        Node == node()]; +	    ...</pre> +	</section>    </section>    <section> diff --git a/lib/common_test/install.sh.in b/lib/common_test/install.sh.in deleted file mode 100644 index 5108c7a259..0000000000 --- a/lib/common_test/install.sh.in +++ /dev/null @@ -1,53 +0,0 @@ -#!/bin/sh - -CT_ROOT=$1 -CT_VSN=@CT_VSN@ -TS_VSN=@TS_VSN@ - -if [ -z "$CT_ROOT" ] -then -    echo "install.sh: need CT_ROOT (absolute) directory or 'local' as argument" -    exit 1 -fi - -if [ $CT_ROOT = "local" ] -then -    CT_DIR=`pwd` -    cd priv -    sed -e "s,@CTPATH@,$CT_DIR/ebin," \ -        -e "s,@TSPATH@,$CT_DIR/../test_server/ebin," \ -	run_test.in > bin/run_test -    chmod 775 bin/run_test -    echo "install successful, start script created in " $CT_ROOT/common_test-$CT_VSN/priv/bin -else - -    if [ ! -d "$CT_ROOT" ] -    then -	echo "install.sh: CT_ROOT argument must be a valid directory" -	exit 1 -    fi - -    if [ `echo $CT_ROOT | awk '{ print substr($1,1,1) }'` != "/" ] -    then -	echo "install.sh: need an absolute path to CT_ROOT" -	exit 1 -    fi - -    if [ ! -d $CT_ROOT/common_test-$CT_VSN ] -    then -	echo "install.sh: The directory $CT_ROOT/common_test-$CT_VSN does not exist" -	exit 1 -    fi - -    if [ -d $CT_ROOT/common_test-$CT_VSN/priv ] -    then -	cd $CT_ROOT/common_test-$CT_VSN/priv -	sed -e "s;@CTPATH@;$CT_ROOT/common_test-$CT_VSN/ebin;" \ -	    -e "s;@TSPATH@;$CT_ROOT/test_server-$TS_VSN/ebin;" \ -	    run_test.in > bin/run_test -	chmod 775 bin/run_test -	echo "install successful, start script created in " $CT_ROOT/common_test-$CT_VSN/priv/bin -    fi -fi - - diff --git a/lib/common_test/priv/Makefile.in b/lib/common_test/priv/Makefile.in index 1bc6b82ebb..7765b06f95 100644 --- a/lib/common_test/priv/Makefile.in +++ b/lib/common_test/priv/Makefile.in @@ -66,12 +66,7 @@ JS = jquery-latest.js jquery.tablesorter.min.js  # Rules  # -include ../../test_server/vsn.mk  debug opt: -	$(V_at)sed -e 's;@CT_VSN@;$(VSN);' \ -            -e 's;@TS_VSN@;$(TEST_SERVER_VSN);' \ -            ../install.sh.in > install.sh -	- $(V_at)chmod -f 775 install.sh  docs: diff --git a/lib/common_test/priv/bin/.gitignore b/lib/common_test/priv/bin/.gitignore deleted file mode 100644 index e69de29bb2..0000000000 --- a/lib/common_test/priv/bin/.gitignore +++ /dev/null diff --git a/lib/common_test/priv/run_test.in b/lib/common_test/priv/run_test.in deleted file mode 100644 index 1508751e4f..0000000000 --- a/lib/common_test/priv/run_test.in +++ /dev/null @@ -1,63 +0,0 @@ -#!/bin/sh  - -args="" - -while [ $1 ]; do -    if [ $1 = "-config" ]; then -	args="$args -ct_config"; -    elif [ $1 = "-decrypt_key" ]; then -	args="$args -ct_decrypt_key"; -    elif [ $1 = "-decrypt_file" ]; then -	args="$args -ct_decrypt_file"; -    elif [ $1 = "-vts" ]; then -	vts=1; -	args="$args $1"; -    elif [ $1 = "-browser" ]; then -	browser=$2; -	args="$args $1"; -    elif [ $1 = "-shell" ]; then -	shell=1; -	args="$args $1"; -    elif [ $1 = "-ctname" ]; then -	ctname=$2; -	args="$args"; -    elif [ $1 = "-ctmaster" ]; then -	master=1; -	args="$args"; -    else -	args="$args $1" -    fi -    shift -done - -if [ $vts ]; then -    erl -sname ct \ -	-pa @CTPATH@ \ -	-pa @TSPATH@ \ -	-s webtool script_start vts $browser \ -	-s ct_run script_start \ -	$args; -elif [ $shell ]; then -    erl -sname ct \ -	-pa @CTPATH@ \ -	-pa @TSPATH@ \ -	-s ct_run script_start \ -	$args; -elif [ $ctname ]; then -    erl -sname $ctname \ -	-pa @CTPATH@ \ -	-pa @TSPATH@ \ -	$args; -elif [ $master ]; then -    erl -sname ct_master \ -	-pa @CTPATH@ \ -	-pa @TSPATH@ \ -	$args; -else -    erl -sname ct \ -	-pa @CTPATH@ \ -	-pa @TSPATH@ \ -	-s ct_run script_start \ -	-s erlang halt \ -	$args -fi diff --git a/lib/common_test/src/Makefile b/lib/common_test/src/Makefile index 8d74546880..e3d5102db8 100644 --- a/lib/common_test/src/Makefile +++ b/lib/common_test/src/Makefile @@ -1,7 +1,7 @@  #  # %CopyrightBegin%  # -# Copyright Ericsson AB 2003-2013. All Rights Reserved. +# Copyright Ericsson AB 2003-2014. 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 @@ -62,6 +62,8 @@ MODULES= \  	ct_telnet_client \  	ct_make \  	vts \ +	ct_webtool \ +	ct_webtool_sup \  	unix_telnet \  	ct_config \  	ct_config_plain \ @@ -75,7 +77,8 @@ MODULES= \  	ct_conn_log_h \  	cth_conn_log \  	ct_groups \ -	ct_property_test +	ct_property_test \ +	ct_release_test  TARGET_MODULES= $(MODULES:%=$(EBIN)/%)  BEAM_FILES= $(MODULES:%=$(EBIN)/%.$(EMULATOR)) diff --git a/lib/common_test/src/common_test.app.src b/lib/common_test/src/common_test.app.src index 580d5dbd7b..0be1466fc9 100644 --- a/lib/common_test/src/common_test.app.src +++ b/lib/common_test/src/common_test.app.src @@ -63,9 +63,10 @@  		ct_master_logs]},    {applications, [kernel,stdlib]},    {env, []}, -  {runtime_dependencies,["xmerl-1.3.7","webtool-0.8.10","tools-2.6.14", -			 "test_server-3.7.1","stdlib-2.0","ssh-3.0.1", -			 "snmp-4.25.1","sasl-2.4","runtime_tools-1.8.14", -			 "kernel-3.0","inets-5.10","erts-6.0", -			 "debugger-4.0","crypto-3.3","compiler-5.0"]}]}. +  {runtime_dependencies,["xmerl-1.3.8","tools-2.8", +			 "test_server-3.9","stdlib-2.5","ssh-4.0", +			 "snmp-5.1.2","sasl-2.4.2","runtime_tools-1.8.16", +			 "kernel-4.0","inets-6.0","erts-7.0", +			 "debugger-4.1","crypto-3.6","compiler-6.0", +			 "observer-2.1"]}]}. diff --git a/lib/common_test/src/ct.erl b/lib/common_test/src/ct.erl index 9d8fce2789..5ed1346f1e 100644 --- a/lib/common_test/src/ct.erl +++ b/lib/common_test/src/ct.erl @@ -79,6 +79,7 @@  %% Other interface functions  -export([get_status/0, abort_current_testcase/1,  	 get_event_mgr_ref/0, +	 get_testspec_terms/0, get_testspec_terms/1,  	 encrypt_config_file/2, encrypt_config_file/3,  	 decrypt_config_file/2, decrypt_config_file/3]). @@ -463,6 +464,50 @@ reload_config(Required)->      ct_config:reload_config(Required).  %%%----------------------------------------------------------------- +%%% @spec get_testspec_terms() -> TestSpecTerms | undefined +%%%      TestSpecTerms = [{Tag,Value}] +%%%      Value = [term()] +%%% +%%% @doc Get a list of all test specification terms used to +%%% configure and run this test. +%%% +get_testspec_terms() -> +    case ct_util:get_testdata(testspec) of +	undefined -> +	    undefined; +	CurrSpecRec -> +	    ct_testspec:testspec_rec2list(CurrSpecRec) +    end. + +%%%----------------------------------------------------------------- +%%% @spec get_testspec_terms(Tags) -> TestSpecTerms | undefined +%%%      Tags = [Tag] | Tag +%%%      Tag = atom() +%%%      TestSpecTerms = [{Tag,Value}] | {Tag,Value} +%%%      Value = [{Node,term()}] | [term()] +%%%      Node = atom() +%%% +%%% @doc Read one or more terms from the test specification used +%%% to configure and run this test. Tag is any valid test specification +%%% tag, such as e.g. <c>label</c>, <c>config</c>, <c>logdir</c>. +%%% User specific terms are also available to read if the +%%% <c>allow_user_terms</c> option has been set. Note that all value tuples +%%% returned, except user terms, will have the node name as first element. +%%% Note also that in order to read test terms, use <c>Tag = tests</c> +%%% (rather than <c>suites</c>, <c>groups</c> or <c>cases</c>). Value is +%%% then the list of *all* tests on the form: +%%% <c>[{Node,Dir,[{TestSpec,GroupsAndCases1},...]},...], where +%%% GroupsAndCases = [{Group,[Case]}] | [Case]</c>. +get_testspec_terms(Tags) -> +    case ct_util:get_testdata(testspec) of +	undefined -> +	    undefined; +	CurrSpecRec -> +	    ct_testspec:testspec_rec2list(Tags, CurrSpecRec) +    end. + + +%%%-----------------------------------------------------------------  %%% @spec log(Format) -> ok  %%% @equiv log(default,50,Format,[])  log(Format) -> diff --git a/lib/common_test/src/ct_config.erl b/lib/common_test/src/ct_config.erl index 5c80a299f8..4b92ca6f8f 100644 --- a/lib/common_test/src/ct_config.erl +++ b/lib/common_test/src/ct_config.erl @@ -693,8 +693,7 @@ make_crypto_key(String) ->      {[K1,K2,K3],IVec}.  random_bytes(N) -> -    {A,B,C} = now(), -    random:seed(A, B, C), +    random:seed(os:timestamp()),      random_bytes_1(N, []).  random_bytes_1(0, Acc) -> Acc; diff --git a/lib/common_test/src/ct_conn_log_h.erl b/lib/common_test/src/ct_conn_log_h.erl index cff02a46d9..2d15035cd8 100644 --- a/lib/common_test/src/ct_conn_log_h.erl +++ b/lib/common_test/src/ct_conn_log_h.erl @@ -34,6 +34,8 @@  -define(WIDTH,80). +-define(now, os:timestamp()). +  %%%-----------------------------------------------------------------  %%% Callbacks  init({GL,ConnLogs}) -> @@ -72,14 +74,14 @@ handle_event({_Type, GL, _Msg}, State) when node(GL) /= node() ->  handle_event({_Type,GL,{Pid,{ct_connection,Mod,Action,ConnName},Report}},  	     State) ->      Info = conn_info(Pid,#conn_log{name=ConnName,action=Action,module=Mod}), -    write_report(now(),Info,Report,GL,State), +    write_report(?now,Info,Report,GL,State),      {ok, State};  handle_event({_Type,GL,{Pid,Info=#conn_log{},Report}}, State) -> -    write_report(now(),conn_info(Pid,Info),Report,GL,State), +    write_report(?now,conn_info(Pid,Info),Report,GL,State),      {ok, State};  handle_event({error_report,GL,{Pid,_,[{ct_connection,ConnName}|R]}}, State) ->      %% Error reports from connection -    write_error(now(),conn_info(Pid,#conn_log{name=ConnName}),R,GL,State), +    write_error(?now,conn_info(Pid,#conn_log{name=ConnName}),R,GL,State),      {ok, State};  handle_event(_What, State) ->      {ok, State}. diff --git a/lib/common_test/src/ct_framework.erl b/lib/common_test/src/ct_framework.erl index 498950c9d1..91368d3137 100644 --- a/lib/common_test/src/ct_framework.erl +++ b/lib/common_test/src/ct_framework.erl @@ -113,6 +113,7 @@ init_tc1(?MODULE,_,error_in_suite,[Config0]) when is_list(Config0) ->      ct_event:notify(#event{name=tc_start,  			   node=node(),  			   data={?MODULE,error_in_suite}}), +    ct_suite_init(?MODULE, error_in_suite, [], Config0),      case ?val(error, Config0) of  	undefined ->  	    {fail,"unknown_error_in_suite"}; @@ -635,7 +636,20 @@ try_set_default(Name,Key,Info,Where) ->  end_tc(Mod, Fun, Args) ->      %% Have to keep end_tc/3 for backwards compatibility issues      end_tc(Mod, Fun, Args, '$end_tc_dummy'). -end_tc(?MODULE,error_in_suite,_, _) ->		% bad start! +end_tc(?MODULE,error_in_suite,{Result,[Args]},Return) -> +    %% this clause gets called if CT has encountered a suite that +    %% can't be executed +    FinalNotify = +	case ct_hooks:end_tc(?MODULE, error_in_suite, Args, Result, Return) of +	    '$ct_no_change' -> +		Result; +	    HookResult -> +		HookResult +	end, +    Event = #event{name=tc_done, +		   node=node(), +		   data={?MODULE,error_in_suite,tag(FinalNotify)}}, +    ct_event:sync_notify(Event),      ok;  end_tc(Mod,Func,{TCPid,Result,[Args]}, Return) when is_pid(TCPid) ->      end_tc(Mod,Func,TCPid,Result,Args,Return); @@ -873,8 +887,8 @@ error_notification(Mod,Func,_Args,{Error,Loc}) ->      end,      PrintErr = fun(ErrFormat, ErrArgs) -> -		       Div = "~n- - - - - - - - - - - - - - - - " -			   "- - - - - - - - - -~n", +		       Div = "~n- - - - - - - - - - - - - - - - - - - " +			     "- - - - - - - - - - - - - - - - - - - - -~n",  		       io:format(user, lists:concat([Div,ErrFormat,Div,"~n"]),  				 ErrArgs),  		       Link = @@ -1062,9 +1076,32 @@ get_all_cases1(_, []) ->  get_all(Mod, ConfTests) ->	      case catch apply(Mod, all, []) of -	{'EXIT',_} -> -	    Reason =  -		list_to_atom(atom_to_list(Mod)++":all/0 is missing"), +	{'EXIT',{undef,[{Mod,all,[],_} | _]}} -> +	    Reason = +		case code:which(Mod) of +		    non_existing -> +			list_to_atom(atom_to_list(Mod)++ +					 " can not be compiled or loaded"); +		    _ -> +			list_to_atom(atom_to_list(Mod)++":all/0 is missing") +		end, +	    %% this makes test_server call error_in_suite as first +	    %% (and only) test case so we can report Reason properly +	    [{?MODULE,error_in_suite,[[{error,Reason}]]}]; +	{'EXIT',ExitReason} -> +	    case ct_util:get_testdata({error_in_suite,Mod}) of +		undefined -> +		    ErrStr = io_lib:format("~n*** ERROR *** " +					   "~w:all/0 failed: ~p~n", +					   [Mod,ExitReason]), +		    io:format(user, ErrStr, []), +		    %% save the error info so it doesn't get printed twice +		    ct_util:set_testdata_async({{error_in_suite,Mod}, +						ExitReason}); +		_ExitReason -> +		    ct_util:delete_testdata({error_in_suite,Mod}) +	    end, +	    Reason = list_to_atom(atom_to_list(Mod)++":all/0 failed"),  	    %% this makes test_server call error_in_suite as first  	    %% (and only) test case so we can report Reason properly  	    [{?MODULE,error_in_suite,[[{error,Reason}]]}]; @@ -1287,6 +1324,8 @@ report(What,Data) ->  	    end,  	    ct_logs:unregister_groupleader(ReportingPid),  	    case {Func,Result} of +		{error_in_suite,_} when Suite == ?MODULE -> +		    ok;  		{init_per_suite,_} ->  		    ok;  		{end_per_suite,_} -> diff --git a/lib/common_test/src/ct_gen_conn.erl b/lib/common_test/src/ct_gen_conn.erl index 56082086f6..8da10ee0f3 100644 --- a/lib/common_test/src/ct_gen_conn.erl +++ b/lib/common_test/src/ct_gen_conn.erl @@ -24,10 +24,9 @@  -module(ct_gen_conn). --compile(export_all). - --export([start/4, stop/1, get_conn_pid/1]). +-export([start/4, stop/1, get_conn_pid/1, check_opts/1]).  -export([call/2, call/3, return/2, do_within_time/2]). +-export([log/3, start_log/1, cont_log/2, end_log/0]).  %%----------------------------------------------------------------------  %% Exported types diff --git a/lib/common_test/src/ct_logs.erl b/lib/common_test/src/ct_logs.erl index dc118ed149..7c8c720e13 100644 --- a/lib/common_test/src/ct_logs.erl +++ b/lib/common_test/src/ct_logs.erl @@ -73,6 +73,8 @@  -define(abs(Name), filename:absname(Name)). +-define(now, os:timestamp()). +  -record(log_cache, {version,  		    all_runs = [],  		    tests = []}). @@ -311,7 +313,7 @@ unregister_groupleader(Pid) ->  %%% data to log (as in <code>io:format(Format,Args)</code>).</p>  log(Heading,Format,Args) ->      cast({log,sync,self(),group_leader(),ct_internal,?MAX_IMPORTANCE, -	  [{int_header(),[log_timestamp(now()),Heading]}, +	  [{int_header(),[log_timestamp(?now),Heading]},  	   {Format,Args},  	   {int_footer(),[]}]}),      ok. @@ -333,7 +335,7 @@ log(Heading,Format,Args) ->  %%% @see end_log/0  start_log(Heading) ->      cast({log,sync,self(),group_leader(),ct_internal,?MAX_IMPORTANCE, -	  [{int_header(),[log_timestamp(now()),Heading]}]}), +	  [{int_header(),[log_timestamp(?now),Heading]}]}),      ok.  %%%----------------------------------------------------------------- @@ -491,11 +493,11 @@ tc_print(Category,Importance,Format,Args) ->  get_heading(default) ->      io_lib:format("\n-----------------------------"  		  "-----------------------\n~s\n", -		  [log_timestamp(now())]); +		  [log_timestamp(?now)]);  get_heading(Category) ->      io_lib:format("\n-----------------------------"  		  "-----------------------\n~s  ~w\n", -		  [log_timestamp(now()),Category]).     +		  [log_timestamp(?now),Category]).      %%%----------------------------------------------------------------- @@ -553,13 +555,13 @@ div_header(Class) ->      div_header(Class,"User").  div_header(Class,Printer) ->      "\n<div class=\"" ++ atom_to_list(Class) ++ "\"><b>*** " ++ Printer ++ -    " " ++ log_timestamp(now()) ++ " ***</b>". +    " " ++ log_timestamp(?now) ++ " ***</b>".  div_footer() ->      "</div>".  maybe_log_timestamp() -> -    {MS,S,US} = now(), +    {MS,S,US} = ?now,      case get(log_timestamp) of  	{MS,S,_} ->  	    ok; @@ -686,7 +688,7 @@ logger(Parent, Mode, Verbosity) ->      make_last_run_index(Time),      CtLogFd = open_ctlog(?misc_io_log),      io:format(CtLogFd,int_header()++int_footer(), -	      [log_timestamp(now()),"Common Test Logger started"]), +	      [log_timestamp(?now),"Common Test Logger started"]),      Parent ! {started,self(),{Time,filename:absname("")}},      set_evmgr_gl(CtLogFd), @@ -835,7 +837,7 @@ logger_loop(State) ->  	stop ->  	    io:format(State#logger_state.ct_log_fd,  		      int_header()++int_footer(), -		      [log_timestamp(now()),"Common Test Logger finished"]), +		      [log_timestamp(?now),"Common Test Logger finished"]),  	    close_ctlog(State#logger_state.ct_log_fd),  	    ok      end. @@ -1908,13 +1910,14 @@ sort_all_runs(Dirs) ->  sort_ct_runs(Dirs) ->      %% Directory naming: <Prefix>.NodeName.Date_Time[/...]      %% Sort on Date_Time string: "YYYY-MM-DD_HH.MM.SS" -    lists:sort(fun(Dir1,Dir2) -> -		       [_Prefix,_Node1,DateHH1,MM1,SS1] = -			   string:tokens(filename:dirname(Dir1),[$.]), -		       [_Prefix,_Node2,DateHH2,MM2,SS2] = -			   string:tokens(filename:dirname(Dir2),[$.]), -		       {DateHH1,MM1,SS1} =< {DateHH2,MM2,SS2} -	       end, Dirs). +    lists:sort( +      fun(Dir1,Dir2) -> +	      [SS1,MM1,DateHH1 | _] = +		  lists:reverse(string:tokens(filename:dirname(Dir1),[$.])), +	      [SS2,MM2,DateHH2 | _] = +		  lists:reverse(string:tokens(filename:dirname(Dir2),[$.])), +	      {DateHH1,MM1,SS1} =< {DateHH2,MM2,SS2} +      end, Dirs).  dir_diff_all_runs(Dirs, LogCache) ->      case LogCache#log_cache.all_runs of @@ -2051,6 +2054,13 @@ runentry(Dir, Totals={Node,Label,Logs,  					     ?testname_width-3)),  		lists:flatten(io_lib:format("~ts...",[Trunc]))  	end, +    TotMissingStr = +	if NotBuilt > 0 -> +		["<font color=\"red\">", +		 integer_to_list(NotBuilt),"</font>"]; +	   true -> +		integer_to_list(NotBuilt) +	end,      Total = TotSucc+TotFail+AllSkip,      A = xhtml(["<td align=center><font size=\"-1\">",Node,  	       "</font></td>\n", @@ -2070,7 +2080,7 @@ runentry(Dir, Totals={Node,Label,Logs,  	 "<td align=right>",TotFailStr,"</td>\n",  	 "<td align=right>",integer_to_list(AllSkip),  	 " (",UserSkipStr,"/",AutoSkipStr,")</td>\n", -	 "<td align=right>",integer_to_list(NotBuilt),"</td>\n"], +	 "<td align=right>",TotMissingStr,"</td>\n"],      TotalsStr = A++B++C,      XHTML = [xhtml("<tr>\n", ["<tr class=\"",odd_or_even(),"\">\n"]), diff --git a/lib/common_test/src/ct_master_logs.erl b/lib/common_test/src/ct_master_logs.erl index 5393097f57..384c1f6863 100644 --- a/lib/common_test/src/ct_master_logs.erl +++ b/lib/common_test/src/ct_master_logs.erl @@ -37,6 +37,8 @@  -define(details_file_name,"details.info").  -define(table_color,"lightblue"). +-define(now, os:timestamp()). +  %%%--------------------------------------------------------------------  %%% API  %%%-------------------------------------------------------------------- @@ -54,7 +56,7 @@ start(LogDir,Nodes) ->      end.  log(Heading,Format,Args) -> -    cast({log,self(),[{int_header(),[log_timestamp(now()),Heading]}, +    cast({log,self(),[{int_header(),[log_timestamp(?now),Heading]},  		      {Format,Args},  		      {int_footer(),[]}]}),      ok. @@ -132,7 +134,7 @@ init(Parent,LogDir,Nodes) ->  					atom_to_list(N) ++ " "  				end,Nodes)), -    io:format(CtLogFd,int_header(),[log_timestamp(now()),"Test Nodes\n"]), +    io:format(CtLogFd,int_header(),[log_timestamp(?now),"Test Nodes\n"]),      io:format(CtLogFd,"~ts\n",[NodeStr]),      io:put_chars(CtLogFd,[int_footer(),"\n"]), @@ -189,7 +191,7 @@ loop(State) ->  	    make_all_runs_index(State#state.logdir),  	    io:format(State#state.log_fd,  		      int_header()++int_footer(), -		      [log_timestamp(now()),"Finished!"]), +		      [log_timestamp(?now),"Finished!"]),  	    close_ct_master_log(State#state.log_fd),  	    close_nodedir_index(State#state.nodedir_ix_fd),  	    ok diff --git a/lib/common_test/src/ct_netconfc.erl b/lib/common_test/src/ct_netconfc.erl index 85fb1ea8d2..cca08bd063 100644 --- a/lib/common_test/src/ct_netconfc.erl +++ b/lib/common_test/src/ct_netconfc.erl @@ -172,6 +172,7 @@  	 only_open/2,  	 hello/1,  	 hello/2, +	 hello/3,  	 close_session/1,  	 close_session/2,  	 kill_session/2, @@ -190,6 +191,7 @@  	 get_config/4,  	 edit_config/3,  	 edit_config/4, +	 edit_config/5,  	 delete_config/2,  	 delete_config/3,  	 copy_config/3, @@ -456,23 +458,35 @@ only_open(KeyOrName, ExtraOpts) ->  %%----------------------------------------------------------------------  %% @spec hello(Client) -> Result -%% @equiv hello(Client, infinity) +%% @equiv hello(Client, [], infinity)  hello(Client) -> -    hello(Client,?DEFAULT_TIMEOUT). +    hello(Client,[],?DEFAULT_TIMEOUT).  %%----------------------------------------------------------------------  -spec hello(Client,Timeout) -> Result when        Client :: handle(),        Timeout :: timeout(),        Result :: ok | {error,error_reason()}. +%% @spec hello(Client, Timeout) -> Result +%% @equiv hello(Client, [], Timeout) +hello(Client,Timeout) -> +    hello(Client,[],Timeout). + +%%---------------------------------------------------------------------- +-spec hello(Client,Options,Timeout) -> Result when +      Client :: handle(), +      Options :: [{capability, [string()]}], +      Timeout :: timeout(), +      Result :: ok | {error,error_reason()}.  %% @doc Exchange `hello' messages with the server.  %% -%% Sends a `hello' message to the server and waits for the return. -%% +%% Adds optional capabilities and sends a `hello' message to the +%% server and waits for the return.  %% @end  %%---------------------------------------------------------------------- -hello(Client,Timeout) -> -    call(Client, {hello, Timeout}). +hello(Client,Options,Timeout) -> +    call(Client, {hello, Options, Timeout}). +  %%----------------------------------------------------------------------  %% @spec get_session_id(Client) -> Result @@ -678,15 +692,39 @@ get_config(Client, Source, Filter, Timeout) ->  %%----------------------------------------------------------------------  %% @spec edit_config(Client, Target, Config) -> Result -%% @equiv edit_config(Client, Target, Config, infinity) +%% @equiv edit_config(Client, Target, Config, [], infinity)  edit_config(Client, Target, Config) ->      edit_config(Client, Target, Config, ?DEFAULT_TIMEOUT).  %%---------------------------------------------------------------------- --spec edit_config(Client, Target, Config, Timeout) -> Result when +-spec edit_config(Client, Target, Config, OptParamsOrTimeout) -> Result when        Client :: client(),        Target :: netconf_db(),        Config :: simple_xml(), +      OptParamsOrTimeout :: [simple_xml()] | timeout(), +      Result :: ok | {error,error_reason()}. +%% @doc +%% +%% If `OptParamsOrTimeout' is a timeout value, then this is +%% equivalent to {@link edit_config/5. edit_config(Client, Target, +%% Config, [], Timeout)}. +%% +%% If `OptParamsOrTimeout' is a list of simple XML, then this is +%% equivalent to {@link edit_config/5. edit_config(Client, Target, +%% Config, OptParams, infinity)}. +%% +%% @end +edit_config(Client, Target, Config, Timeout) when ?is_timeout(Timeout) -> +    edit_config(Client, Target, Config, [], Timeout); +edit_config(Client, Target, Config, OptParams) when is_list(OptParams) -> +    edit_config(Client, Target, Config, OptParams, ?DEFAULT_TIMEOUT). + +%%---------------------------------------------------------------------- +-spec edit_config(Client, Target, Config, OptParams, Timeout) -> Result when +      Client :: client(), +      Target :: netconf_db(), +      Config :: simple_xml(), +      OptParams :: [simple_xml()],        Timeout :: timeout(),        Result :: ok | {error,error_reason()}.  %% @doc Edit configuration data. @@ -695,10 +733,20 @@ edit_config(Client, Target, Config) ->  %% include `:candidate' or `:startup' in its list of  %% capabilities.  %% +%% `OptParams' can be used for specifying optional parameters +%% (`default-operation', `test-option' or `error-option') that will be +%% added to the `edit-config' request. The value must be a list +%% containing valid simple XML, for example +%% +%% ``` +%% [{'default-operation', ["none"]}, +%%  {'error-option', ["rollback-on-error"]}] +%%''' +%%  %% @end  %%---------------------------------------------------------------------- -edit_config(Client, Target, Config, Timeout) -> -    call(Client, {send_rpc_op, edit_config, [Target,Config], Timeout}). +edit_config(Client, Target, Config, OptParams, Timeout) -> +    call(Client, {send_rpc_op, edit_config, [Target,Config,OptParams], Timeout}).  %%---------------------------------------------------------------------- @@ -1040,9 +1088,9 @@ terminate(_, #state{connection=Connection}) ->      ok.  %% @private -handle_msg({hello,Timeout}, From, +handle_msg({hello, Options, Timeout}, From,  	   #state{connection=Connection,hello_status=HelloStatus} = State) -> -    case do_send(Connection, client_hello()) of +    case do_send(Connection, client_hello(Options)) of  	ok ->  	    case HelloStatus of  		undefined -> @@ -1087,6 +1135,7 @@ handle_msg({get_event_streams=Op,Streams,Timeout}, From, State) ->      SimpleXml = encode_rpc_operation(get,[Filter]),      do_send_rpc(Op, SimpleXml, Timeout, From, State). +%% @private  handle_msg({ssh_cm, CM, {data, Ch, _Type, Data}}, State) ->      ssh_connection:adjust_window(CM,Ch,size(Data)),      handle_data(Data, State); @@ -1118,7 +1167,9 @@ handle_msg({Ref,timeout},#state{pending=Pending} = State) ->  	    close_session -> stop;  	    _ -> noreply  	end, -    {R,State#state{pending=Pending1}}. +    %% Halfhearted try to get in correct state, this matches +    %% the implementation before this patch +    {R,State#state{pending=Pending1, buff= <<>>}}.  %% @private  %% Called by ct_util_server to close registered connections before terminate. @@ -1222,10 +1273,14 @@ set_request_timer(T) ->  %%%----------------------------------------------------------------- -client_hello() -> +client_hello(Options) when is_list(Options) -> +    UserCaps = [{capability, UserCap} || +		   {capability, UserCap} <- Options, +		   is_list(hd(UserCap))],      {hello, ?NETCONF_NAMESPACE_ATTR,       [{capabilities, -       [{capability,[?NETCONF_BASE_CAP++?NETCONF_BASE_CAP_VSN]}]}]}. +       [{capability,[?NETCONF_BASE_CAP++?NETCONF_BASE_CAP_VSN]}| +	UserCaps]}]}.  %%%----------------------------------------------------------------- @@ -1235,8 +1290,8 @@ encode_rpc_operation(get,[Filter]) ->      {get,filter(Filter)};  encode_rpc_operation(get_config,[Source,Filter]) ->      {'get-config',[{source,[Source]}] ++ filter(Filter)}; -encode_rpc_operation(edit_config,[Target,Config]) -> -    {'edit-config',[{target,[Target]},{config,[Config]}]}; +encode_rpc_operation(edit_config,[Target,Config,OptParams]) -> +    {'edit-config',[{target,[Target]}] ++ OptParams ++ [{config,[Config]}]};  encode_rpc_operation(delete_config,[Target]) ->      {'delete-config',[{target,[Target]}]};  encode_rpc_operation(copy_config,[Target,Source]) -> @@ -1308,72 +1363,54 @@ to_xml_doc(Simple) ->  %%%-----------------------------------------------------------------  %%% Parse and handle received XML data -handle_data(NewData,#state{connection=Connection,buff=Buff} = State) -> +handle_data(NewData,#state{connection=Connection,buff=Buff0} = State0) ->      log(Connection,recv,NewData), -    Data = <<Buff/binary,NewData/binary>>, -    case xmerl_sax_parser:stream(<<>>, -				 [{continuation_fun,fun sax_cont/1}, -				  {continuation_state,{Data,Connection,false}}, -				  {event_fun,fun sax_event/3}, -				  {event_state,[]}]) of -	{ok, Simple, Rest} -> -	    decode(Simple,State#state{buff=Rest}); -	{fatal_error,_Loc,Reason,_EndTags,_EventState} -> -	    ?error(Connection#connection.name,[{parse_error,Reason}, -					       {buffer,Buff}, -					       {new_data,NewData}]), -	    case Reason of -		{could_not_fetch_data,Msg} -> -		    handle_msg(Msg,State#state{buff = <<>>}); -		_Other -> -		    Pending1 = -			case State#state.pending of -			    [] -> -				[]; -			    Pending -> -				%% Assuming the first request gets the -				%% first answer -				P=#pending{tref=TRef,caller=Caller} = -				    lists:last(Pending), -				_ = timer:cancel(TRef), -				Reason1 = {failed_to_parse_received_data,Reason}, -				ct_gen_conn:return(Caller,{error,Reason1}), -				lists:delete(P,Pending) -			end, -		    {noreply,State#state{pending=Pending1,buff = <<>>}} -	    end -    end. - -%%%----------------------------------------------------------------- -%%% Parsing of XML data -%% Contiuation function for the sax parser -sax_cont(done) -> -    {<<>>,done}; -sax_cont({Data,Connection,false}) -> +    Data = append_wo_initial_nl(Buff0,NewData),      case binary:split(Data,[?END_TAG],[]) of -	[All] -> -	    %% No end tag found. Remove what could be a part -	    %% of an end tag from the data and save for next -	    %% iteration -	    SafeSize = size(All)-5, -	    <<New:SafeSize/binary,Save:5/binary>> = All, -	    {New,{Save,Connection,true}}; -	[_Msg,_Rest]=Msgs -> -	    %% We have at least one full message. Any excess data will -	    %% be returned from xmerl_sax_parser:stream/2 in the Rest -	    %% parameter. -	    {list_to_binary(Msgs),done} -    end; -sax_cont({Data,Connection,true}) -> -    case ssh_receive_data() of -	{ok,Bin} -> -	    log(Connection,recv,Bin), -	    sax_cont({<<Data/binary,Bin/binary>>,Connection,false}); -	{error,Reason} -> -	    throw({could_not_fetch_data,Reason}) +	[_NoEndTagFound] -> +	    {noreply, State0#state{buff=Data}}; +	[FirstMsg,Buff1] -> +	    SaxArgs = [{event_fun,fun sax_event/3}, {event_state,[]}], +	    case xmerl_sax_parser:stream(FirstMsg, SaxArgs) of +		{ok, Simple, _Thrash} -> +		    case decode(Simple, State0#state{buff=Buff1}) of +			{noreply, #state{buff=Buff} = State} when Buff =/= <<>> -> +			    %% Recurse if we have more data in buffer +			    handle_data(<<>>, State); +			Other -> +			    Other +		    end; +		{fatal_error,_Loc,Reason,_EndTags,_EventState} -> +		    ?error(Connection#connection.name, +			   [{parse_error,Reason}, +			    {buffer, Buff0}, +			    {new_data,NewData}]), +		    handle_error(Reason, State0#state{buff= <<>>}) +	    end      end. - +%% xml does not accept a leading nl and some netconf server add a nl after +%% each ?END_TAG, ignore them +append_wo_initial_nl(<<>>,NewData) -> NewData; +append_wo_initial_nl(<<"\n", Data/binary>>, NewData) -> +    append_wo_initial_nl(Data, NewData); +append_wo_initial_nl(Data, NewData) -> +    <<Data/binary, NewData/binary>>. + +handle_error(Reason, State) -> +    Pending1 = case State#state.pending of +		   [] -> []; +		   Pending -> +		       %% Assuming the first request gets the +		       %% first answer +		       P=#pending{tref=TRef,caller=Caller} = +			   lists:last(Pending), +		       _ = timer:cancel(TRef), +		       Reason1 = {failed_to_parse_received_data,Reason}, +		       ct_gen_conn:return(Caller,{error,Reason1}), +		       lists:delete(P,Pending) +	       end, +    {noreply, State#state{pending=Pending1}}.  %% Event function for the sax parser. It builds a simple XML structure.  %% Care is taken to keep namespace attributes and prefixes as in the original XML. @@ -1711,6 +1748,7 @@ log(#connection{host=Host,port=Port,name=Name},Action,Data) ->  %% Log callback - called from the error handler process +%% @private  format_data(How,Data) ->      %% Assuming that the data is encoded as UTF-8.  If it is not, then      %% the printout might be wrong, but the format function will not @@ -1836,16 +1874,6 @@ get_tag([]) ->  %%%-----------------------------------------------------------------  %%% SSH stuff -ssh_receive_data() -> -    receive -	{ssh_cm, CM, {data, Ch, _Type, Data}} -> -	    ssh_connection:adjust_window(CM,Ch,size(Data)), -	    {ok, Data}; -        {ssh_cm, _CM, {Closed, _Ch}} = X when Closed == closed; Closed == eof -> -            {error,X}; -	{_Ref,timeout} = X -> -	    {error,X} -    end.  ssh_open(#options{host=Host,timeout=Timeout,port=Port,ssh=SshOpts,name=Name}) ->      case ssh:connect(Host, Port, diff --git a/lib/common_test/src/ct_release_test.erl b/lib/common_test/src/ct_release_test.erl new file mode 100644 index 0000000000..3f0b5bda67 --- /dev/null +++ b/lib/common_test/src/ct_release_test.erl @@ -0,0 +1,936 @@ +%% +%% %CopyrightBegin% +%% +%% Copyright Ericsson AB 2014-2015. 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% +%% +%%----------------------------------------------------------------- +%% @doc EXPERIMENTAL support for testing of upgrade. +%% +%% This is a library module containing support for test of release +%% related activities in one or more applications. Currenty it +%% supports upgrade only. +%% +%% == Configuration == +%% +%% In order to find version numbers of applications to upgrade from, +%% `{@module}' needs to access and start old OTP +%% releases. A `common_test' configuration file can be used for +%% specifying the location of such releases, for example: +%% +%% ``` +%% %% old-rels.cfg +%% {otp_releases,[{r16b,"/path/to/R16B03-1/bin/erl"}, +%% 	       {'17',"/path/to/17.3/bin/erl"}]}.''' +%% +%% The configuration file should preferably point out the latest patch +%% level on each major release. +%% +%% If no such configuration file is given, {@link init/1} will return +%% `{skip,Reason}' and any attempt at running {@link upgrade/4} +%% will fail. +%% +%% == Callback functions == +%% +%% The following functions should be exported from a {@module} +%% callback module. +%% +%% All callback functions are called on the node where the upgrade is +%% executed. +%% +%% <dl> +%%   <dt>Module:upgrade_init(CtData,State) -> NewState</dt> +%%   <dd>Types: +%% +%%     <b><code>CtData = {@link ct_data()}</code></b><br/> +%%     <b><code>State = NewState = cb_state()</code></b> +%% +%%     Initialyze system before upgrade test starts. +%% +%%     This function is called before the upgrade is started. All +%%     applications given in {@link upgrade/4} are already started by +%%     the boot script, so this callback is intended for additional +%%     initialization, if necessary. +%% +%%     <code>CtData</code> is an opaque data structure which shall be used +%%     in any call to <code>ct_release_test</code> inside the callback. +%% +%%     Example: +%% +%% ``` +%% upgrade_init(CtData,State) -> +%%     {ok,{FromVsn,ToVsn}} = ct_release_test:get_app_vsns(CtData,myapp), +%%     open_connection(State).''' +%%   </dd> +%% +%%   <dt>Module:upgrade_upgraded(CtData,State) -> NewState</dt> +%%   <dd>Types: +%% +%%     <b><code>CtData = {@link ct_data()}</code></b><br/> +%%     <b><code>State = NewState = cb_state()</code></b> +%% +%%     Check that upgrade was successful. +%% +%%     This function is called after the release_handler has +%%     successfully unpacked and installed the new release, and it has +%%     been made permanent. It allows application specific checks to +%%     ensure that the upgrade was successful. +%% +%%     <code>CtData</code> is an opaque data structure which shall be used +%%     in any call to <code>ct_release_test</code> inside the callback. +%% +%%     Example: +%% +%% ``` +%% upgrade_upgraded(CtData,State) -> +%%     check_connection_still_open(State).''' +%%   </dd> +%% +%%   <dt>Module:upgrade_downgraded(CtData,State) -> NewState</dt> +%%   <dd>Types: +%% +%%     <b><code>CtData = {@link ct_data()}</code></b><br/> +%%     <b><code>State = NewState = cb_state()</code></b> +%% +%%     Check that downgrade was successful. +%% +%%     This function is called after the release_handler has +%%     successfully re-installed the original release, and it has been +%%     made permanent. It allows application specific checks to ensure +%%     that the downgrade was successful. +%% +%%     <code>CtData</code> is an opaque data structure which shall be used +%%     in any call to <code>ct_release_test</code> inside the callback. +%% +%%     Example: +%% +%% ``` +%% upgrade_downgraded(CtData,State) -> +%%     check_connection_closed(State).''' +%%   </dd> +%% </dl> +%% @end +%%----------------------------------------------------------------- +-module(ct_release_test). + +-export([init/1, upgrade/4, cleanup/1, get_app_vsns/2, get_appup/2]). + +-include_lib("kernel/include/file.hrl"). + +%%----------------------------------------------------------------- +-define(testnode, otp_upgrade). +-define(exclude_apps, [hipe, typer, dialyzer]). % never include these apps + +%%----------------------------------------------------------------- +-record(ct_data, {from,to}). + +%%----------------------------------------------------------------- +-type config() :: [{atom(),term()}]. +-type cb_state() :: term(). +-opaque ct_data() :: #ct_data{}. +-export_type([ct_data/0]). + +-callback upgrade_init(ct_data(),cb_state()) -> cb_state(). +-callback upgrade_upgraded(ct_data(),cb_state()) -> cb_state(). +-callback upgrade_downgraded(ct_data(),cb_state()) -> cb_state(). + +%%----------------------------------------------------------------- +-spec init(Config) -> Result when +      Config :: config(), +      Result :: config() | SkipOrFail, +      SkipOrFail :: {skip,Reason} | {fail,Reason}. +%% @doc Initialize `{@module}'. +%% +%% This function can be called from any of the +%% `init_per_*' functions in the test suite. It updates +%% the given `Config' with data that will be +%% used by future calls to other functions in this module. The +%% returned configuration must therefore also be returned from +%% the calling `init_per_*'. +%% +%% If the initialization fails, e.g. if a required release can +%% not be found, the function returns `{skip,Reason}'. In +%% this case the other test support functions in this mudule +%% can not be used. +%% +%% Example: +%% +%% ``` +%% init_per_suite(Config) -> +%%     ct_release_test:init(Config).''' +%% +init(Config) -> +    try init_upgrade_test() of +	{Major,Minor} -> +	    [{release_test,[{major,Major},{minor,Minor}]} | Config] +    catch throw:Thrown -> +	    Thrown +    end. + +%%----------------------------------------------------------------- +-spec upgrade(App,Level,Callback,Config) -> any() when +      App :: atom(), +      Level :: minor | major, +      Callback :: {module(),InitState}, +      InitState :: cb_state(), +      Config :: config(); +	     (Apps,Level,Callback,Config) -> any() when +      Apps :: [App], +      App :: atom(), +      Level :: minor | major, +      Callback :: {module(),InitState}, +      InitState :: cb_state(), +      Config :: config(). +%% @doc Test upgrade of the given application(s). +%% +%% This function can be called from a test case. It requires that +%% `Config' has been initialized by calling {@link +%% init/1} prior to this, for example from `init_per_suite/1'. +%% +%% Upgrade tests are performed as follows: +%% +%% <ol> +%%   <li>Figure out which OTP release to test upgrade +%%     from. Start a node running that release and find the +%%     application versions on that node. Terminate the +%%     node.</li> +%%   <li>Figure out all dependencies for the applications under +%%     test.</li> +%%   <li>Create a release containing the core +%%     applications `kernel', `stdlib' and `sasl' +%%     in addition to the application(s) under test and all +%%     dependencies of these. The versions of the applications +%%     under test will be the ones found on the OTP release to +%%     upgrade from. The versions of all other applications will +%%     be those found on the current node, i.e. the common_test +%%     node. This is the "From"-release.</li> +%%   <li>Create another release containing the same +%%     applications as in the previous step, but with all +%%     application versions taken from the current node. This is +%%     the "To"-release.</li> +%%   <li>Install the "From"-release and start a new node +%%     running this release.</li> +%%   <li>Perform the upgrade test and allow customized +%%     control by using callbacks: +%%     <ol> +%%       <li>Callback: `upgrade_init/2'</li> +%%       <li>Unpack the new release</li> +%%       <li>Install the new release</li> +%%       <li>Callback: `upgrade_upgraded/2'</li> +%%       <li>Install the original release</li> +%%       <li>Callback: `upgrade_downgraded/2'</li> +%%     </ol> +%%   </li> +%% </ol> +%% +%% `App' or `Apps' +%% specifies the applications under test, i.e. the applications +%% which shall be upgraded. All other applications that are +%% included have the same releases in the "From"- and +%% "To"-releases and will therefore not be upgraded. +%% +%% `Level' specifies which OTP release to +%% pick the "From" versions from. +%% <dl> +%%   <dt>major</dt> +%%   <dd>From verions are picked from the previous major +%%     release. For example, if the test is run on an OTP-17 +%%     node, `{@module}' will pick the application +%%     "From" versions from an OTP installation running OTP +%%     R16B.</dd> +%% +%%   <dt>minor</dt> +%%   <dd>From verions are picked from the current major +%%     release. For example, if the test is run on an OTP-17 +%%     node, `{@module}' will pick the application +%%     "From" versions from an OTP installation running an +%%     earlier patch level of OTP-17.</dd> +%% </dl> +%% +%% The application "To" versions are allways picked from the +%% current node, i.e. the common_test node. +%% +%% `Callback' specifies the module (normally the +%% test suite) which implements the {@section Callback functions}, and +%% the initial value of the `State' variable used in these +%% functions. +%% +%% `Config' is the input argument received +%% in the test case function. +%% +%% Example: +%% +%% ``` +%% minor_upgrade(Config) -> +%%     ct_release_test:upgrade(ssl,minor,{?MODULE,[]},Config). +%% ''' +%% +upgrade(App,Level,Callback,Config) when is_atom(App) -> +    upgrade([App],Level,Callback,Config); +upgrade(Apps,Level,Callback,Config) -> +    Dir = proplists:get_value(priv_dir,Config), +    CreateDir = filename:join([Dir,Level,create]), +    InstallDir = filename:join([Dir,Level,install]), +    ok = filelib:ensure_dir(filename:join(CreateDir,"*")), +    ok = filelib:ensure_dir(filename:join(InstallDir,"*")), +    try upgrade(Apps,Level,Callback,CreateDir,InstallDir,Config) of +	ok -> +	    %%rm_rf(CreateDir), +	    Tars = filelib:wildcard(filename:join(CreateDir,"*.tar.gz")), +	    _ = [file:delete(Tar) || Tar <- Tars], +	    rm_rf(InstallDir), +	    ok +    catch throw:{fail,Reason} -> +	    ct:fail(Reason); +	  throw:{skip,Reason} -> +	    rm_rf(CreateDir), +	    rm_rf(InstallDir), +	    {skip,Reason} +    after +	%% Brutally kill all nodes that erroneously survived the test. +	%% Note, we will not reach this if the test fails with a +	%% timetrap timeout in the test suite! Thus we can have +	%% hanging nodes... +	Nodes = nodes(), +	[rpc:call(Node,erlang,halt,[]) || Node <- Nodes] +    end. + +%%----------------------------------------------------------------- +-spec cleanup(Config) -> Result when +      Config :: config(), +      Result :: config(). +%% @doc Clean up after tests. +%% +%% This function shall be called from the `end_per_*' function +%% complementing the `init_per_*' function where {@link init/1} +%% is called. +%% +%% It cleans up after the test, for example kills hanging +%% nodes. +%% +%% Example: +%% +%% ``` +%% end_per_suite(Config) -> +%%     ct_release_test:cleanup(Config).''' +%% +cleanup(Config) -> +    Nodes = [node_name(?testnode)|nodes()], +    [rpc:call(Node,erlang,halt,[]) || Node <- Nodes], +    Config. + +%%----------------------------------------------------------------- +-spec get_app_vsns(CtData,App) -> {ok,{From,To}} | {error,Reason} when +      CtData :: ct_data(), +      App :: atom(), +      From :: string(), +      To :: string(), +      Reason :: {app_not_found,App}. +%% @doc Get versions involved in this upgrade for the given application. +%% +%% This function can be called from inside any of the callback +%% functions. It returns the old (From) and new (To) versions involved +%% in the upgrade/downgrade test for the given application. +%% +%% <code>CtData</code> must be the first argument received in the +%% calling callback function - an opaque data structure set by +%% <code>ct_release_tests</code>. +get_app_vsns(#ct_data{from=FromApps,to=ToApps},App) -> +    case {lists:keyfind(App,1,FromApps),lists:keyfind(App,1,ToApps)} of +	{{App,FromVsn,_},{App,ToVsn,_}} -> +	    {ok,{FromVsn,ToVsn}}; +	_ -> +	    {error,{app_not_found,App}} +    end. + +%%----------------------------------------------------------------- +-spec get_appup(CtData,App) -> {ok,Appup} | {error,Reason} when +      CtData :: ct_data(), +      App :: atom(), +      Appup :: {From,To,Up,Down}, +      From :: string(), +      To :: string(), +      Up :: [Instr], +      Down :: [Instr], +      Instr :: term(), +      Reason :: {app_not_found,App} | {vsn_not_found,{App,From}}. +%% @doc Get appup instructions for the given application. +%% +%% This function can be called from inside any of the callback +%% functions. It reads the appup file for the given application and +%% returns the instructions for upgrade and downgrade for the versions +%% in the test. +%% +%% <code>CtData</code> must be the first argument received in the +%% calling callback function - an opaque data structure set by +%% <code>ct_release_tests</code>. +%% +%% See reference manual for appup files for types definitions for the +%% instructions. +get_appup(#ct_data{from=FromApps,to=ToApps},App) -> +    case lists:keyfind(App,1,ToApps) of +	{App,ToVsn,ToDir} -> +	    Appup = filename:join([ToDir, "ebin", atom_to_list(App)++".appup"]), +	    {ok, [{ToVsn, Ups, Downs}]} = file:consult(Appup), +	    {App,FromVsn,_} = lists:keyfind(App,1,FromApps), +	    case {systools_relup:appup_search_for_version(FromVsn,Ups), +		  systools_relup:appup_search_for_version(FromVsn,Downs)} of +		{{ok,Up},{ok,Down}} -> +		    {ok,{FromVsn,ToVsn,Up,Down}}; +		_ -> +		    {error,{vsn_not_found,{App,FromVsn}}} +	    end; +	false -> +	    {error,{app_not_found,App}} +    end. + +%%----------------------------------------------------------------- +init_upgrade_test() -> +    %% Check that a real release is running, not e.g. cerl +    ok = application:ensure_started(sasl), +    case release_handler:which_releases() of +	[{_,_,[],_}] -> +	    %% Fake release, no applications +	    throw({skip, "Need a real release running to create other releases"}); +	_ -> +	    Major = init_upgrade_test(major), +	    Minor = init_upgrade_test(minor), +	    {Major,Minor} +    end. + +init_upgrade_test(Level) -> +    {FromVsn,ToVsn} = get_rels(Level), +    OldRel = +	case test_server:is_release_available(FromVsn) of +	    true -> +		{release,FromVsn}; +	    false -> +		case ct:get_config({otp_releases,list_to_atom(FromVsn)}) of +		    undefined -> +			false; +		    Prog0 -> +			case os:find_executable(Prog0) of +			    false -> +				false; +			    Prog -> +				{prog,Prog} +			end +		end +	end, +    case OldRel of +	false -> +	    ct:log("Release ~p is not available." +		   " Upgrade on '~p' level can not be tested.", +		   [FromVsn,Level]), +	    undefined; +	_ -> +	    init_upgrade_test(FromVsn,ToVsn,OldRel) +    end. + +get_rels(major) -> +    %% Given that the current major release is X, then this is an +    %% upgrade from major release X-1 to the current release. +    Current = erlang:system_info(otp_release), +    PreviousMajor = previous_major(Current), +    {PreviousMajor,Current}; +get_rels(minor) -> +    %% Given that this is a (possibly) patched version of major +    %% release X, then this is an upgrade from major release X to the +    %% current release. +    CurrentMajor = erlang:system_info(otp_release), +    Current = CurrentMajor++"_patched", +    {CurrentMajor,Current}. + +init_upgrade_test(FromVsn,ToVsn,OldRel) -> +    OtpRel = list_to_atom("otp-"++FromVsn), +    ct:log("Starting node to fetch application versions to upgrade from"), +    {ok,Node} = test_server:start_node(OtpRel,peer,[{erl,[OldRel]}]), +    {Apps,Path} = fetch_all_apps(Node), +    test_server:stop_node(Node), +    {FromVsn,ToVsn,Apps,Path}. + +fetch_all_apps(Node) -> +    Paths = rpc:call(Node,code,get_path,[]), +    %% Find all possible applications in the path +    AppFiles = +	lists:flatmap( +	  fun(P) -> +		  filelib:wildcard(filename:join(P,"*.app")) +	  end, +	  Paths), +    %% Figure out which version of each application is running on this +    %% node. Using application:load and application:get_key instead of +    %% reading the .app files since there might be multiple versions +    %% of a .app file and we only want the one that is actually +    %% running. +    AppVsns = +	lists:flatmap( +	  fun(F) -> +		  A = list_to_atom(filename:basename(filename:rootname(F))), +		  _ = rpc:call(Node,application,load,[A]), +		  case rpc:call(Node,application,get_key,[A,vsn]) of +		      {ok,V} -> [{A,V}]; +		      _ -> [] +		  end +	  end, +	  AppFiles), +    ErtsVsn = rpc:call(Node, erlang, system_info, [version]), +    {[{erts,ErtsVsn}|AppVsns], Paths}. + + +%%----------------------------------------------------------------- +upgrade(Apps,Level,Callback,CreateDir,InstallDir,Config) -> +    ct:log("Test upgrade of the following applications: ~p",[Apps]), +    ct:log(".rel files and start scripts are created in:~n~ts",[CreateDir]), +    ct:log("The release is installed in:~n~ts",[InstallDir]), +    case proplists:get_value(release_test,Config) of +	undefined -> +	    throw({fail,"ct_release_test:init/1 not run"}); +	RTConfig -> +	    case proplists:get_value(Level,RTConfig) of +		undefined -> +		    throw({skip,"Old release not available"}); +		Data -> +		    {FromVsn,FromRel,FromAppsVsns} = +			target_system(Apps, CreateDir, InstallDir, Data), +		    {ToVsn,ToRel,ToAppsVsns} = +			upgrade_system(Apps, FromRel, CreateDir, +				       InstallDir, Data), +		    ct:log("Upgrade from: OTP-~ts, ~p",[FromVsn, FromAppsVsns]), +		    ct:log("Upgrade to: OTP-~ts, ~p",[ToVsn, ToAppsVsns]), +		    do_upgrade(Callback, FromVsn, FromAppsVsns, ToRel, +			       ToAppsVsns, InstallDir) +	    end +    end. + +%%% This is similar to sasl/examples/src/target_system.erl, but with +%%% the following adjustments: +%%% - add a log directory +%%% - use an own 'start' script +%%% - chmod 'start' and 'start_erl' +target_system(Apps,CreateDir,InstallDir,{FromVsn,_,AllAppsVsns,Path}) -> +    RelName0 = "otp-"++FromVsn, + +    AppsVsns = [{A,V} || {A,V} <- AllAppsVsns, lists:member(A,Apps)], +    {RelName,ErtsVsn} = create_relfile(AppsVsns,CreateDir,RelName0,FromVsn), + +    %% Create .script and .boot +    ok = systools(make_script,[RelName,[{path,Path}]]), + +    %% Create base tar file - i.e. erts and all apps +    ok = systools(make_tar,[RelName,[{erts,code:root_dir()}, +				     {path,Path}]]), + +    %% Unpack the tar to complete the installation +    erl_tar:extract(RelName ++ ".tar.gz", [{cwd, InstallDir}, compressed]), + +    %% Add bin and log dirs +    BinDir = filename:join([InstallDir, "bin"]), +    file:make_dir(BinDir), +    file:make_dir(filename:join(InstallDir,"log")), + +    %% Delete start scripts - they will be added later +    ErtsBinDir = filename:join([InstallDir, "erts-" ++ ErtsVsn, "bin"]), +    file:delete(filename:join([ErtsBinDir, "erl"])), +    file:delete(filename:join([ErtsBinDir, "start"])), +    file:delete(filename:join([ErtsBinDir, "start_erl"])), + +    %% Copy .boot to bin/start.boot +    copy_file(RelName++".boot",filename:join([BinDir, "start.boot"])), + +    %% Copy scripts from erts-xxx/bin to bin +    copy_file(filename:join([ErtsBinDir, "epmd"]), +              filename:join([BinDir, "epmd"]), [preserve]), +    copy_file(filename:join([ErtsBinDir, "run_erl"]), +              filename:join([BinDir, "run_erl"]), [preserve]), +    copy_file(filename:join([ErtsBinDir, "to_erl"]), +              filename:join([BinDir, "to_erl"]), [preserve]), + +    %% create start_erl.data, sys.config and start.src +    StartErlData = filename:join([InstallDir, "releases", "start_erl.data"]), +    write_file(StartErlData, io_lib:fwrite("~s ~s~n", [ErtsVsn, FromVsn])), +    SysConfig = filename:join([InstallDir, "releases", FromVsn, "sys.config"]), +    write_file(SysConfig, "[]."), +    StartSrc = filename:join(ErtsBinDir,"start.src"), +    write_file(StartSrc,start_script()), +    ok = file:change_mode(StartSrc,8#0755), + +    %% Make start_erl executable +    %% (this has been fixed in OTP 17 - it is now installed with +    %% $INSTALL_SCRIPT instead of $INSTALL_DATA and should therefore +    %% be executable from the start) +    ok = file:change_mode(filename:join(ErtsBinDir,"start_erl.src"),8#0755), + +    %% Substitute variables in erl.src, start.src and start_erl.src +    %% (.src found in erts-xxx/bin - result stored in bin) +    subst_src_scripts(["erl", "start", "start_erl"], ErtsBinDir, BinDir, +                      [{"FINAL_ROOTDIR", InstallDir}, {"EMU", "beam"}], +                      [preserve]), + +    %% Create RELEASES +    RelFile = filename:join([InstallDir, "releases", +			     filename:basename(RelName) ++ ".rel"]), +    release_handler:create_RELEASES(InstallDir, RelFile), + +    {FromVsn, RelName,AppsVsns}. + +systools(Func,Args) -> +    case apply(systools,Func,Args) of +	ok -> +	    ok; +	error -> +	    throw({fail,{systools,Func,Args}}) +    end. + +%%% This is a copy of $ROOT/erts-xxx/bin/start.src, modified to add +%%% sname and heart +start_script() -> +    ["#!/bin/sh\n" +     "ROOTDIR=%FINAL_ROOTDIR%\n" +     "\n" +     "if [ -z \"$RELDIR\" ]\n" +     "then\n" +     "   RELDIR=$ROOTDIR/releases\n" +     "fi\n" +     "\n" +     "START_ERL_DATA=${1:-$RELDIR/start_erl.data}\n" +     "\n" +     "$ROOTDIR/bin/run_erl -daemon /tmp/ $ROOTDIR/log \"exec $ROOTDIR/bin/start_erl $ROOTDIR $RELDIR $START_ERL_DATA -sname ",atom_to_list(?testnode)," -heart\"\n"]. + +%%% Create a release containing the current (the test node) OTP +%%% release, including relup to allow upgrade from an earlier OTP +%%% release. +upgrade_system(Apps, FromRel, CreateDir, InstallDir, {_,ToVsn,_,_}) -> +    ct:log("Generating release to upgrade to."), + +    RelName0 = "otp-"++ToVsn, + +    AppsVsns = get_vsns(Apps), +    {RelName,_} = create_relfile(AppsVsns,CreateDir,RelName0,ToVsn), +    FromPath = filename:join([InstallDir,lib,"*",ebin]), + +    ok = systools(make_script,[RelName]), +    ok = systools(make_relup,[RelName,[FromRel],[FromRel], +			      [{path,[FromPath]}, +			       {outdir,CreateDir}]]), +    SysConfig = filename:join([CreateDir, "sys.config"]), +    write_file(SysConfig, "[]."), + +    ok = systools(make_tar,[RelName,[{erts,code:root_dir()}]]), + +    {ToVsn, RelName,AppsVsns}. + +%%% Start a new node running the release from target_system/6 +%%% above. Then upgrade to the system from upgrade_system/6. +do_upgrade({Cb,InitState},FromVsn,FromAppsVsns,ToRel,ToAppsVsns,InstallDir) -> +    ct:log("Upgrade test attempting to start node.~n" +	   "If test fails, logs can be found in:~n~ts", +	   [filename:join(InstallDir,log)]), +    Start = filename:join([InstallDir,bin,start]), +    {ok,Node} = start_node(Start,FromVsn,FromAppsVsns), + +    %% Add path to this module, to allow calls to get_appup/2 +    Dir = filename:dirname(code:which(?MODULE)), +    _ = rpc:call(Node,code,add_pathz,[Dir]), + +    ct:log("Node started: ~p",[Node]), +    CtData = #ct_data{from = [{A,V,code:lib_dir(A)} || {A,V} <- FromAppsVsns], +		      to=[{A,V,code:lib_dir(A)} || {A,V} <- ToAppsVsns]}, +    State1 = do_callback(Node,Cb,upgrade_init,[CtData,InitState]), + +    [{"OTP upgrade test",FromVsn,_,permanent}] = +	rpc:call(Node,release_handler,which_releases,[]), +    ToRelName = filename:basename(ToRel), +    copy_file(ToRel++".tar.gz", +	      filename:join([InstallDir,releases,ToRelName++".tar.gz"])), +    ct:log("Unpacking new release"), +    {ok,ToVsn} = rpc:call(Node,release_handler,unpack_release,[ToRelName]), +    [{"OTP upgrade test",ToVsn,_,unpacked}, +     {"OTP upgrade test",FromVsn,_,permanent}] = +	rpc:call(Node,release_handler,which_releases,[]), +    ct:log("Installing new release"), +    case rpc:call(Node,release_handler,install_release,[ToVsn]) of +	{ok,FromVsn,_} -> +	    ok; +	{continue_after_restart,FromVsn,_} -> +	    ct:log("Waiting for node restart") +    end, +    %% even if install_release returned {ok,...} there might be an +    %% emulator restart (instruction restart_emulator), so we must +    %% always make sure the node is running. +    wait_node_up(current,ToVsn,ToAppsVsns), + +    [{"OTP upgrade test",ToVsn,_,current}, +     {"OTP upgrade test",FromVsn,_,permanent}] = +	rpc:call(Node,release_handler,which_releases,[]), +    ct:log("Permanenting new release"), +    ok = rpc:call(Node,release_handler,make_permanent,[ToVsn]), +    [{"OTP upgrade test",ToVsn,_,permanent}, +     {"OTP upgrade test",FromVsn,_,old}] = +	rpc:call(Node,release_handler,which_releases,[]), + +    State2 = do_callback(Node,Cb,upgrade_upgraded,[CtData,State1]), + +    ct:log("Re-installing old release"), +    case rpc:call(Node,release_handler,install_release,[FromVsn]) of +	{ok,FromVsn,_} -> +	    ok; +	{continue_after_restart,FromVsn,_} -> +	    ct:log("Waiting for node restart") +    end, +    %% even if install_release returned {ok,...} there might be an +    %% emulator restart (instruction restart_emulator), so we must +    %% always make sure the node is running. +    wait_node_up(current,FromVsn,FromAppsVsns), + +    [{"OTP upgrade test",ToVsn,_,permanent}, +     {"OTP upgrade test",FromVsn,_,current}] = +	rpc:call(Node,release_handler,which_releases,[]), +    ct:log("Permanenting old release"), +    ok = rpc:call(Node,release_handler,make_permanent,[FromVsn]), +    [{"OTP upgrade test",ToVsn,_,old}, +     {"OTP upgrade test",FromVsn,_,permanent}] = +	rpc:call(Node,release_handler,which_releases,[]), + +    _State3 = do_callback(Node,Cb,upgrade_downgraded,[CtData,State2]), + +    ct:log("Terminating node ~p",[Node]), +    erlang:monitor_node(Node,true), +    _ = rpc:call(Node,init,stop,[]), +    receive {nodedown,Node} -> ok end, +    ct:log("Node terminated"), + +    ok. + +do_callback(Node,Mod,Func,Args) -> +    Dir = filename:dirname(code:which(Mod)), +    _ = rpc:call(Node,code,add_path,[Dir]), +    ct:log("Calling ~p:~p/1",[Mod,Func]), +    R = rpc:call(Node,Mod,Func,Args), +    ct:log("~p:~p/~w returned: ~p",[Mod,Func,length(Args),R]), +    case R of +	{badrpc,Error} -> +	    test_server:fail({test_upgrade_callback,Mod,Func,Args,Error}); +	NewState -> +	    NewState +    end. + +%%% Library functions +previous_major("17") -> +    "r16b"; +previous_major(Rel) -> +    integer_to_list(list_to_integer(Rel)-1). + +create_relfile(AppsVsns,CreateDir,RelName0,RelVsn) -> +    UpgradeAppsVsns = [{A,V,restart_type(A)} || {A,V} <- AppsVsns], + +    CoreAppVsns0 = get_vsns([kernel,stdlib,sasl]), +    CoreAppVsns = +	[{A,V,restart_type(A)} || {A,V} <- CoreAppVsns0, +				  false == lists:keymember(A,1,AppsVsns)], + +    Apps = [App || {App,_} <- AppsVsns], +    StartDepsVsns = get_start_deps(Apps,CoreAppVsns), +    StartApps = [StartApp || {StartApp,_,_} <- StartDepsVsns] ++ Apps, + +    {RuntimeDepsVsns,_} = get_runtime_deps(StartApps,StartApps,[],[]), + +    AllAppsVsns0 = StartDepsVsns ++ UpgradeAppsVsns ++ RuntimeDepsVsns, + +    %% Should test tools really be included? Some library functions +    %% here could be used by callback, but not everything since +    %% processes of these applications will not be running. +    TestToolAppsVsns0 = get_vsns([test_server,common_test]), +    TestToolAppsVsns = +	[{A,V,none} || {A,V} <- TestToolAppsVsns0, +		       false == lists:keymember(A,1,AllAppsVsns0)], + +    AllAppsVsns1 = AllAppsVsns0 ++ TestToolAppsVsns, +    AllAppsVsns = [AV || AV={A,_,_} <- AllAppsVsns1, +			 false == lists:member(A,?exclude_apps)], + +    ErtsVsn = erlang:system_info(version), + +    %% Create the .rel file +    RelContent = {release,{"OTP upgrade test",RelVsn},{erts,ErtsVsn},AllAppsVsns}, +    RelName = filename:join(CreateDir,RelName0), +    RelFile = RelName++".rel", +    {ok,Fd} = file:open(RelFile,[write,{encoding,utf8}]), +    io:format(Fd,"~tp.~n",[RelContent]), +    ok = file:close(Fd), +    {RelName,ErtsVsn}. + +get_vsns(Apps) -> +    [begin +	 _ = application:load(A), +	 {ok,V} = application:get_key(A,vsn), +	 {A,V} +     end || A <- Apps]. + +get_start_deps([App|Apps],Acc) -> +    _ = application:load(App), +    {ok,StartDeps} = application:get_key(App,applications), +    StartDepsVsns = +	[begin +	     _ = application:load(StartApp), +	     {ok,StartVsn} = application:get_key(StartApp,vsn), +	     {StartApp,StartVsn,restart_type(StartApp)} +	 end || StartApp <- StartDeps, +		false == lists:keymember(StartApp,1,Acc)], +    DepsStartDeps = get_start_deps(StartDeps,Acc ++ StartDepsVsns), +    get_start_deps(Apps,DepsStartDeps); +get_start_deps([],Acc) -> +    Acc. + +get_runtime_deps([App|Apps],StartApps,Acc,Visited) -> +    case lists:member(App,Visited) of +	true -> +	    get_runtime_deps(Apps,StartApps,Acc,Visited); +	false -> +	    %% runtime_dependencies should be possible to read with +	    %% application:get_key/2, but still isn't so we need to +	    %% read the .app file... +	    AppFile = code:where_is_file(atom_to_list(App) ++ ".app"), +	    {ok,[{application,App,Attrs}]} = file:consult(AppFile), +	    RuntimeDeps = +		lists:flatmap( +		  fun(Str) -> +			  [RuntimeAppStr,_] = string:tokens(Str,"-"), +			  RuntimeApp = list_to_atom(RuntimeAppStr), +			  case {lists:keymember(RuntimeApp,1,Acc), +				lists:member(RuntimeApp,StartApps)} of +			      {false,false} when RuntimeApp=/=erts -> +				  [RuntimeApp]; +			      _ -> +				  [] +			  end +		  end, +		  proplists:get_value(runtime_dependencies,Attrs,[])), +	    RuntimeDepsVsns = +		[begin +		     _ = application:load(RuntimeApp), +		     {ok,RuntimeVsn} = application:get_key(RuntimeApp,vsn), +		     {RuntimeApp,RuntimeVsn,none} +		 end || RuntimeApp <- RuntimeDeps], +	    {DepsRuntimeDeps,NewVisited} = +		get_runtime_deps(RuntimeDeps,StartApps,Acc++RuntimeDepsVsns,[App|Visited]), +	    get_runtime_deps(Apps,StartApps,DepsRuntimeDeps,NewVisited) +    end; +get_runtime_deps([],_,Acc,Visited) -> +    {Acc,Visited}. + +restart_type(App) when App==kernel; App==stdlib; App==sasl -> +    permanent; +restart_type(_) -> +    temporary. + +copy_file(Src, Dest) -> +    copy_file(Src, Dest, []). + +copy_file(Src, Dest, Opts) -> +    {ok,_} = file:copy(Src, Dest), +    case lists:member(preserve, Opts) of +        true -> +            {ok, FileInfo} = file:read_file_info(Src), +            file:write_file_info(Dest, FileInfo); +        false -> +            ok +    end. + +write_file(FName, Conts) -> +    Enc = file:native_name_encoding(), +    {ok, Fd} = file:open(FName, [write]), +    file:write(Fd, unicode:characters_to_binary(Conts,Enc,Enc)), +    file:close(Fd). + +%% Substitute all occurrences of %Var% for Val in the given scripts +subst_src_scripts(Scripts, SrcDir, DestDir, Vars, Opts) -> +    lists:foreach(fun(Script) -> +                          subst_src_script(Script, SrcDir, DestDir, +                                           Vars, Opts) +                  end, Scripts). + +subst_src_script(Script, SrcDir, DestDir, Vars, Opts) -> +    subst_file(filename:join([SrcDir, Script ++ ".src"]), +               filename:join([DestDir, Script]), +               Vars, Opts). + +subst_file(Src, Dest, Vars, Opts) -> +    {ok, Bin} = file:read_file(Src), +    Conts = binary_to_list(Bin), +    NConts = subst(Conts, Vars), +    write_file(Dest, NConts), +    case lists:member(preserve, Opts) of +        true -> +            {ok, FileInfo} = file:read_file_info(Src), +            file:write_file_info(Dest, FileInfo); +        false -> +            ok +    end. + +subst(Str, [{Var,Val}|Vars]) -> +    subst(re:replace(Str,"%"++Var++"%",Val,[{return,list}]),Vars); +subst(Str, []) -> +    Str. + +%%% Start a node by executing the given start command. This node will +%%% be used for upgrade. +start_node(Start,ExpVsn,ExpAppsVsns) -> +    Port = open_port({spawn_executable, Start}, []), +    unlink(Port), +    erlang:port_close(Port), +    wait_node_up(permanent,ExpVsn,ExpAppsVsns). + +wait_node_up(ExpStatus,ExpVsn,ExpAppsVsns) -> +    Node = node_name(?testnode), +    wait_node_up(Node,ExpStatus,ExpVsn,lists:keysort(1,ExpAppsVsns),60). + +wait_node_up(Node,ExpStatus,ExpVsn,ExpAppsVsns,0) -> +    test_server:fail({node_not_started,app_check_failed,ExpVsn,ExpAppsVsns, +		      rpc:call(Node,release_handler,which_releases,[ExpStatus]), +		      rpc:call(Node,application,which_applications,[])}); +wait_node_up(Node,ExpStatus,ExpVsn,ExpAppsVsns,N) -> +    case {rpc:call(Node,release_handler,which_releases,[ExpStatus]), +	  rpc:call(Node, application, which_applications, [])} of +	{[{_,ExpVsn,_,_}],Apps} when is_list(Apps) -> +	    case [{A,V} || {A,_,V} <- lists:keysort(1,Apps), +			   lists:keymember(A,1,ExpAppsVsns)] of +		ExpAppsVsns -> +		    {ok,Node}; +		_ -> +		    timer:sleep(2000), +		    wait_node_up(Node,ExpStatus,ExpVsn,ExpAppsVsns,N-1) +	    end; +	_ -> +	    timer:sleep(2000), +	    wait_node_up(Node,ExpStatus,ExpVsn,ExpAppsVsns,N-1) +    end. + +node_name(Sname) -> +    {ok,Host} = inet:gethostname(), +    list_to_atom(atom_to_list(Sname) ++ "@" ++ Host). + +rm_rf(Dir) -> +    case file:read_file_info(Dir) of +	{ok, #file_info{type = directory}} -> +	    {ok, Content} = file:list_dir_all(Dir), +	    [rm_rf(filename:join(Dir,C)) || C <- Content], +	    ok=file:del_dir(Dir), +	    ok; +	{ok, #file_info{}} -> +	    ok=file:delete(Dir); +	_ -> +	    ok +    end. diff --git a/lib/common_test/src/ct_run.erl b/lib/common_test/src/ct_run.erl index 4a12481214..0eafe72020 100644 --- a/lib/common_test/src/ct_run.erl +++ b/lib/common_test/src/ct_run.erl @@ -77,7 +77,8 @@  	       multiply_timetraps = 1,  	       scale_timetraps = false,  	       create_priv_dir, -	       testspecs = [], +	       testspec_files = [], +	       current_testspec,  	       tests,  	       starter}). @@ -225,18 +226,24 @@ finish(Tracing, ExitStatus, Args) ->      if ExitStatus == interactive_mode ->  	    interactive_mode;         true -> -	    %% it's possible to tell CT to finish execution with a call -	    %% to a different function than the normal halt/1 BIF -	    %% (meant to be used mainly for reading the CT exit status) -	    case get_start_opt(halt_with, -			       fun([HaltMod,HaltFunc]) ->  -				       {list_to_atom(HaltMod), -					list_to_atom(HaltFunc)} end, -			       Args) of -		undefined -> -		    halt(ExitStatus); -		{M,F} -> -		    apply(M, F, [ExitStatus]) +	    case get_start_opt(vts, true, Args) of +		true -> +		    %% VTS mode, don't halt the node +		    ok; +		_ -> +		    %% it's possible to tell CT to finish execution with a call +		    %% to a different function than the normal halt/1 BIF +		    %% (meant to be used mainly for reading the CT exit status) +		    case get_start_opt(halt_with, +				       fun([HaltMod,HaltFunc]) ->  +					       {list_to_atom(HaltMod), +						list_to_atom(HaltFunc)} end, +				       Args) of +			undefined -> +			    halt(ExitStatus); +			{M,F} -> +			    apply(M, F, [ExitStatus]) +		    end  	    end      end. @@ -244,7 +251,7 @@ script_start1(Parent, Args) ->      %% read general start flags      Label = get_start_opt(label, fun([Lbl]) -> Lbl end, Args),      Profile = get_start_opt(profile, fun([Prof]) -> Prof end, Args), -    Vts = get_start_opt(vts, true, Args), +    Vts = get_start_opt(vts, true, undefined, Args),      Shell = get_start_opt(shell, true, Args),      Cover = get_start_opt(cover, fun([CoverFile]) -> ?abs(CoverFile) end, Args),      CoverStop = get_start_opt(cover_stop,  @@ -330,8 +337,8 @@ script_start1(Parent, Args) ->      Stylesheet = get_start_opt(stylesheet,  			       fun([SS]) -> ?abs(SS) end, Args),      %% basic_html - used by ct_logs -    BasicHtml = case proplists:get_value(basic_html, Args) of -		    undefined -> +    BasicHtml = case {Vts,proplists:get_value(basic_html, Args)} of +		    {undefined,undefined} ->  			application:set_env(common_test, basic_html, false),  			undefined;  		    _ -> @@ -364,9 +371,10 @@ script_start1(Parent, Args) ->  		 scale_timetraps = ScaleTT,  		 create_priv_dir = CreatePrivDir,  		 starter = script}, -     +      %% check if log files should be refreshed or go on to run tests...      Result = run_or_refresh(Opts, Args), +      %% send final results to starting process waiting in script_start/0      Parent ! {self(), Result}. @@ -485,8 +493,11 @@ execute_one_spec(TS, Opts, Args) ->      case check_and_install_configfiles(AllConfig, TheLogDir, Opts) of  	ok ->      % read tests from spec  	    {Run,Skip} = ct_testspec:prepare_tests(TS, node()), -	    do_run(Run, Skip, Opts#opts{config=AllConfig, -					logdir=TheLogDir}, Args); +	    Result = do_run(Run, Skip, Opts#opts{config=AllConfig, +						 logdir=TheLogDir, +						 current_testspec=TS}, Args), +	    ct_util:delete_testdata(testspec), +	    Result;  	Error ->  	    Error      end. @@ -577,7 +588,7 @@ combine_test_opts(TS, Specs, Opts) ->      Opts#opts{label = Label,  	      profile = Profile, -	      testspecs = Specs, +	      testspec_files = Specs,  	      cover = Cover,  	      cover_stop = CoverStop,  	      logdir = which(logdir, LogDir), @@ -702,7 +713,7 @@ script_start4(#opts{label = Label, profile = Profile,  		    logopts = LogOpts,  		    verbosity = Verbosity,  		    enable_builtin_hooks = EnableBuiltinHooks, -		    logdir = LogDir, testspecs = Specs}, _Args) -> +		    logdir = LogDir, testspec_files = Specs}, _Args) ->      %% label - used by ct_logs      application:set_env(common_test, test_label, Label), @@ -757,21 +768,6 @@ script_start4(Opts = #opts{tests = Tests}, Args) ->  %%% @doc Print usage information for <code>ct_run</code>.  script_usage() ->      io:format("\n\nUsage:\n\n"), -    io:format("Run tests in web based GUI:\n\n" -	      "\tct_run -vts [-browser Browser]" -	      "\n\t[-config ConfigFile1 ConfigFile2 .. ConfigFileN]" -	      "\n\t[-decrypt_key Key] | [-decrypt_file KeyFile]" -	      "\n\t[-dir TestDir1 TestDir2 .. TestDirN] |" -	      "\n\t[-suite Suite [-case Case]]" -	      "\n\t[-logopts LogOpt1 LogOpt2 .. LogOptN]" -	      "\n\t[-verbosity GenVLvl | [CategoryVLvl1 .. CategoryVLvlN]]" -	      "\n\t[-include InclDir1 InclDir2 .. InclDirN]" -	      "\n\t[-no_auto_compile]" -	      "\n\t[-abort_if_missing_suites]" -	      "\n\t[-multiply_timetraps N]" -	      "\n\t[-scale_timetraps]" -	      "\n\t[-create_priv_dir auto_per_run | auto_per_tc | manual_per_tc]" -	      "\n\t[-basic_html]\n\n"),      io:format("Run tests from command line:\n\n"  	      "\tct_run [-dir TestDir1 TestDir2 .. TestDirN] |"  	      "\n\t[[-dir TestDir] -suite Suite1 Suite2 .. SuiteN" @@ -831,7 +827,22 @@ script_usage() ->      io:format("Run CT in interactive mode:\n\n"  	      "\tct_run -shell"  	      "\n\t[-config ConfigFile1 ConfigFile2 .. ConfigFileN]" -	      "\n\t[-decrypt_key Key] | [-decrypt_file KeyFile]\n\n"). +	      "\n\t[-decrypt_key Key] | [-decrypt_file KeyFile]\n\n"), +    io:format("Run tests in web based GUI:\n\n" +	      "\tct_run -vts [-browser Browser]" +	      "\n\t[-config ConfigFile1 ConfigFile2 .. ConfigFileN]" +	      "\n\t[-decrypt_key Key] | [-decrypt_file KeyFile]" +	      "\n\t[-dir TestDir1 TestDir2 .. TestDirN] |" +	      "\n\t[-suite Suite [-case Case]]" +	      "\n\t[-logopts LogOpt1 LogOpt2 .. LogOptN]" +	      "\n\t[-verbosity GenVLvl | [CategoryVLvl1 .. CategoryVLvlN]]" +	      "\n\t[-include InclDir1 InclDir2 .. InclDirN]" +	      "\n\t[-no_auto_compile]" +	      "\n\t[-abort_if_missing_suites]" +	      "\n\t[-multiply_timetraps N]" +	      "\n\t[-scale_timetraps]" +	      "\n\t[-create_priv_dir auto_per_run | auto_per_tc | manual_per_tc]" +	      "\n\t[-basic_html]\n\n").  %%%-----------------------------------------------------------------  %%% @hidden @@ -1103,7 +1114,7 @@ run_test2(StartOpts) ->  	undefined ->  	    case lists:keysearch(prepared_tests, 1, StartOpts) of  		{value,{_,{Run,Skip},Specs}} ->	% use prepared tests -		    run_prepared(Run, Skip, Opts#opts{testspecs = Specs}, +		    run_prepared(Run, Skip, Opts#opts{testspec_files = Specs},  				 StartOpts);  		false ->  		    run_dir(Opts, StartOpts) @@ -1111,11 +1122,11 @@ run_test2(StartOpts) ->  	Specs ->  	    Relaxed = get_start_opt(allow_user_terms, value, false, StartOpts),  	    %% using testspec(s) as input for test -	    run_spec_file(Relaxed, Opts#opts{testspecs = Specs}, StartOpts) +	    run_spec_file(Relaxed, Opts#opts{testspec_files = Specs}, StartOpts)      end.  run_spec_file(Relaxed, -	      Opts = #opts{testspecs = Specs}, +	      Opts = #opts{testspec_files = Specs},  	      StartOpts) ->      Specs1 = case Specs of  		 [X|_] when is_integer(X) -> [Specs]; @@ -1154,7 +1165,10 @@ run_all_specs([{Specs,TS} | TSs], Opts, StartOpts, TotResult) ->      log_ts_names(Specs),      Combined = #opts{config = TSConfig} = combine_test_opts(TS, Specs, Opts),      AllConfig = merge_vals([Opts#opts.config, TSConfig]), -    try run_one_spec(TS, Combined#opts{config = AllConfig}, StartOpts) of +    try run_one_spec(TS,  +		     Combined#opts{config = AllConfig, +				   current_testspec=TS}, +		     StartOpts) of  	Result ->  	    run_all_specs(TSs, Opts, StartOpts, [Result | TotResult])		      catch @@ -1399,7 +1413,7 @@ run_testspec2(TestSpec) ->  	    case check_and_install_configfiles(  		   Opts#opts.config, LogDir1, Opts) of  		ok -> -		    Opts1 = Opts#opts{testspecs = [], +		    Opts1 = Opts#opts{testspec_files = [],  				      logdir = LogDir1,  				      include = AllInclude},  		    {Run,Skip} = ct_testspec:prepare_tests(TS, node()), @@ -1620,11 +1634,15 @@ groups_and_cases(Gs, Cs) ->  tests(TestDir, Suites, []) when is_list(TestDir), is_integer(hd(TestDir)) ->      [{?testdir(TestDir,Suites),ensure_atom(Suites),all}];  tests(TestDir, Suite, Cases) when is_list(TestDir), is_integer(hd(TestDir)) -> +    [{?testdir(TestDir,Suite),ensure_atom(Suite),Cases}]; +tests([TestDir], Suite, Cases) when is_list(TestDir), is_integer(hd(TestDir)) ->      [{?testdir(TestDir,Suite),ensure_atom(Suite),Cases}].  tests([{Dir,Suite}],Cases) ->      [{?testdir(Dir,Suite),ensure_atom(Suite),Cases}];  tests(TestDir, Suite) when is_list(TestDir), is_integer(hd(TestDir)) -> -    tests(TestDir, ensure_atom(Suite), all). +    tests(TestDir, ensure_atom(Suite), all); +tests([TestDir], Suite) when is_list(TestDir), is_integer(hd(TestDir)) -> +     tests(TestDir, ensure_atom(Suite), all).  tests(DirSuites) when is_list(DirSuites), is_tuple(hd(DirSuites)) ->      [{?testdir(Dir,Suite),ensure_atom(Suite),all} || {Dir,Suite} <- DirSuites];  tests(TestDir) when is_list(TestDir), is_integer(hd(TestDir)) -> @@ -1706,6 +1724,9 @@ compile_and_run(Tests, Skip, Opts, Args) ->      ct_util:set_testdata({stylesheet,Opts#opts.stylesheet}),      %% save logopts      ct_util:set_testdata({logopts,Opts#opts.logopts}), +    %% save info about current testspec (testspec record or undefined) +    ct_util:set_testdata({testspec,Opts#opts.current_testspec}), +      %% enable silent connections      case Opts#opts.silent_connections of  	[] -> @@ -1720,7 +1741,7 @@ compile_and_run(Tests, Skip, Opts, Args) ->  		    ct_logs:log("Silent connections", "~p", [Conns])  	    end      end, -    log_ts_names(Opts#opts.testspecs), +    log_ts_names(Opts#opts.testspec_files),      TestSuites = suite_tuples(Tests),      {_TestSuites1,SuiteMakeErrors,AllMakeErrors} = @@ -1969,22 +1990,7 @@ final_tests(Tests, Skip, Bad) ->  final_tests1([{TestDir,Suites,_}|Tests], Final, Skip, Bad) when        is_list(Suites), is_atom(hd(Suites)) -> -%     Separate = -% 	fun(S,{DoSuite,Dont}) ->		 -% 		case lists:keymember({TestDir,S},1,Bad) of -% 		    false ->	 -% 			{[S|DoSuite],Dont}; -% 		    true ->	 -% 			SkipIt = {TestDir,S,"Make failed"}, -% 			{DoSuite,Dont++[SkipIt]} -% 		end -% 	end, -	 -%     {DoSuites,Skip1} = -% 	lists:foldl(Separate,{[],Skip},Suites), -%     Do = {TestDir,lists:reverse(DoSuites),all}, - -    Skip1 = [{TD,S,"Make failed"} || {{TD,S},_} <- Bad, S1 <- Suites, +    Skip1 = [{TD,S,make_failed} || {{TD,S},_} <- Bad, S1 <- Suites,  				     S == S1, TD == TestDir],      Final1 = [{TestDir,S,all} || S <- Suites],      final_tests1(Tests, lists:reverse(Final1)++Final, Skip++Skip1, Bad); @@ -1997,7 +2003,7 @@ final_tests1([{TestDir,all,all}|Tests], Final, Skip, Bad) ->  	    false ->  		[]  	end, -    Missing = [{TestDir,S,"Make failed"} || S <- MissingSuites], +    Missing = [{TestDir,S,make_failed} || S <- MissingSuites],      Final1 = [{TestDir,all,all}|Final],      final_tests1(Tests, Final1, Skip++Missing, Bad); @@ -2009,7 +2015,7 @@ final_tests1([{TestDir,Suite,GrsOrCs}|Tests], Final, Skip, Bad) when        is_list(GrsOrCs) ->      case lists:keymember({TestDir,Suite}, 1, Bad) of  	true -> -	    Skip1 = Skip ++ [{TestDir,Suite,all,"Make failed"}], +	    Skip1 = Skip ++ [{TestDir,Suite,all,make_failed}],  	    final_tests1(Tests, [{TestDir,Suite,all}|Final], Skip1, Bad);  	false ->  	    GrsOrCs1 = diff --git a/lib/common_test/src/ct_telnet.erl b/lib/common_test/src/ct_telnet.erl index d906a267a1..b14731e74f 100644 --- a/lib/common_test/src/ct_telnet.erl +++ b/lib/common_test/src/ct_telnet.erl @@ -29,7 +29,7 @@  %% Command timeout = 10 sec (time to wait for a command to return)  %% Max no of reconnection attempts = 3  %% Reconnection interval = 5 sek (time to wait in between reconnection attempts) -%% Keep alive = true (will send NOP to the server every 10 sec if connection is idle) +%% Keep alive = true (will send NOP to the server every 8 sec if connection is idle)  %% Polling limit = 0 (max number of times to poll to get a remaining string terminated)  %% Polling interval = 1 sec (sleep time between polls)</pre>  %% <p>These parameters can be altered by the user with the following @@ -486,7 +486,8 @@ expect(Connection,Patterns) ->  %%%      Opts = [Opt]  %%%      Opt = {idle_timeout,IdleTimeout} | {total_timeout,TotalTimeout} |  %%%            repeat | {repeat,N} | sequence | {halt,HaltPatterns} | -%%%            ignore_prompt | no_prompt_check +%%%            ignore_prompt | no_prompt_check | wait_for_prompt | +%%%            {wait_for_prompt,Prompt}  %%%      IdleTimeout = infinity | integer()  %%%      TotalTimeout = infinity | integer()  %%%      N = integer() @@ -499,9 +500,9 @@ expect(Connection,Patterns) ->  %%%  %%% @doc Get data from telnet and wait for the expected pattern.  %%% -%%% <p><code>Pattern</code> can be a POSIX regular expression. If more -%%% than one pattern is given, the function returns when the first -%%% match is found.</p> +%%% <p><code>Pattern</code> can be a POSIX regular expression. The function +%%% returns as soon as a pattern has been successfully matched (at least one, +%%% in the case of multiple patterns).</p>  %%%  %%% <p><code>RxMatch</code> is a list of matched strings. It looks  %%% like this: <code>[FullMatch, SubMatch1, SubMatch2, ...]</code> @@ -524,10 +525,13 @@ expect(Connection,Patterns) ->  %%% milliseconds, <code>{error,timeout}</code> is returned. The default  %%% value is <code>infinity</code> (i.e. no time limit).</p>  %%% -%%% <p>The function will always return when a prompt is found, unless -%%% any of the <code>ignore_prompt</code> or -%%% <code>no_prompt_check</code> options are used, in which case it -%%% will return when a match is found or after a timeout.</p> +%%% <p>The function will return when a prompt is received, even if no +%%% pattern has yet been matched. In this event, +%%% <code>{error,{prompt,Prompt}}</code> is returned. +%%% However, this behaviour may be modified with the +%%% <code>ignore_prompt</code> or <code>no_prompt_check</code> option, which +%%% tells <code>expect</code> to return only when a match is found or after a +%%% timeout.</p>  %%%  %%% <p>If the <code>ignore_prompt</code> option is used,  %%% <code>ct_telnet</code> will ignore any prompt found. This option @@ -541,6 +545,13 @@ expect(Connection,Patterns) ->  %%% is useful if, for instance, the <code>Pattern</code> itself  %%% matches the prompt.</p>  %%% +%%% <p>The <code>wait_for_prompt</code> option forces <code>ct_telnet</code> +%%% to wait until the prompt string has been received before returning +%%% (even if a pattern has already been matched). This is equal to calling: +%%% <code>expect(Conn, Patterns++[{prompt,Prompt}], [sequence|Opts])</code>. +%%% Note that <code>idle_timeout</code> and <code>total_timeout</code> +%%% may abort the operation of waiting for prompt.</p> +%%%  %%% <p>The <code>repeat</code> option indicates that the pattern(s)  %%% shall be matched multiple times. If <code>N</code> is given, the  %%% pattern(s) will be matched <code>N</code> times, and the function @@ -653,18 +664,21 @@ handle_msg({cmd,Cmd,Opts},State) ->      start_gen_log(heading(cmd,State#state.name)),      log(State,cmd,"Cmd: ~p",[Cmd]), +    %% whatever is in the buffer from previous operations +    %% will be ignored as we go ahead with this telnet cmd +      debug_cont_gen_log("Throwing Buffer:",[]),      debug_log_lines(State#state.buffer),      case {State#state.type,State#state.prompt} of -	{ts,_} ->  +	{ts,_} ->  	    silent_teln_expect(State#state.name,  			       State#state.teln_pid,  			       State#state.buffer,  			       prompt,  			       State#state.prx,  			       [{idle_timeout,2000}]); -	{ip,false} ->  +	{ip,false} ->  	    silent_teln_expect(State#state.name,  			       State#state.teln_pid,  			       State#state.buffer, @@ -1007,7 +1021,7 @@ silent_teln_expect(Name,Pid,Data,Pattern,Prx,Opts) ->      put(silent,Old),      Result. -%% teln_expect/5  +%% teln_expect/6  %%  %% This function implements the expect functionality over telnet. In  %% general there are three possible ways to go: @@ -1029,10 +1043,12 @@ teln_expect(Name,Pid,Data,Pattern0,Prx,Opts) ->  	end,      PromptCheck = get_prompt_check(Opts), -    Seq = get_seq(Opts), -    Pattern = convert_pattern(Pattern0,Seq), -    {IdleTimeout,TotalTimeout} = get_timeouts(Opts), +    {WaitForPrompt,Pattern1,Opts1} = wait_for_prompt(Pattern0,Opts), + +    Seq = get_seq(Opts1), +    Pattern2 = convert_pattern(Pattern1,Seq), +    {IdleTimeout,TotalTimeout} = get_timeouts(Opts1),      EO = #eo{teln_pid=Pid,  	     prx=Prx, @@ -1042,9 +1058,16 @@ teln_expect(Name,Pid,Data,Pattern0,Prx,Opts) ->  	     haltpatterns=HaltPatterns,  	     prompt_check=PromptCheck}, -    case get_repeat(Opts) of +    case get_repeat(Opts1) of  	false -> -	    case teln_expect1(Name,Pid,Data,Pattern,[],EO) of +	    case teln_expect1(Name,Pid,Data,Pattern2,[],EO) of +		{ok,Matched,Rest} when WaitForPrompt -> +		    case lists:reverse(Matched) of +			[{prompt,_},Matched1] -> +			    {ok,Matched1,Rest}; +			[{prompt,_}|Matched1] -> +			    {ok,lists:reverse(Matched1),Rest} +		    end;  		{ok,Matched,Rest} ->  		    {ok,Matched,Rest};  		{halt,Why,Rest} -> @@ -1054,7 +1077,7 @@ teln_expect(Name,Pid,Data,Pattern0,Prx,Opts) ->  	    end;  	N ->  	    EO1 = EO#eo{repeat=N}, -	    repeat_expect(Name,Pid,Data,Pattern,[],EO1) +	    repeat_expect(Name,Pid,Data,Pattern2,[],EO1)      end.  convert_pattern(Pattern,Seq)  @@ -1118,6 +1141,40 @@ get_ignore_prompt(Opts) ->  get_prompt_check(Opts) ->      not lists:member(no_prompt_check,Opts). +wait_for_prompt(Pattern, Opts) -> +    case lists:member(wait_for_prompt, Opts) of +	true -> +	    wait_for_prompt1(prompt, Pattern, +			     lists:delete(wait_for_prompt,Opts)); +	false -> +	    case proplists:get_value(wait_for_prompt, Opts) of +		undefined -> +		    {false,Pattern,Opts}; +		PromptStr -> +		    wait_for_prompt1({prompt,PromptStr}, Pattern, +				     proplists:delete(wait_for_prompt,Opts)) +	    end +    end. + +wait_for_prompt1(Prompt, [Ch|_] = Pattern, Opts) when is_integer(Ch) -> +    wait_for_prompt2(Prompt, [Pattern], Opts); +wait_for_prompt1(Prompt, Pattern, Opts) when is_list(Pattern) -> +    wait_for_prompt2(Prompt, Pattern, Opts); +wait_for_prompt1(Prompt, Pattern, Opts) -> +    wait_for_prompt2(Prompt, [Pattern], Opts). + +wait_for_prompt2(Prompt, Pattern, Opts) -> +    Pattern1 = case lists:reverse(Pattern) of +		   [prompt|_]     -> Pattern; +		   [{prompt,_}|_] -> Pattern; +		   _              -> Pattern ++ [Prompt] +	       end, +    Opts1 = case lists:member(sequence, Opts) of +		true ->  Opts; +		false -> [sequence|Opts] +	    end, +    {true,Pattern1,Opts1}. +  %% Repeat either single or sequence. All match results are accumulated  %% and returned when a halt condition is fulllfilled.  repeat_expect(_Name,_Pid,Rest,_Pattern,Acc,#eo{repeat=0}) -> @@ -1210,7 +1267,7 @@ get_data1(Pid) ->  %% 1) Single expect.  %% First the whole data chunk is searched for a prompt (to avoid doing  %% a regexp match for the prompt at each line). -%% If we are searching for anyting else, the datachunk is split into +%% If we are searching for anything else, the datachunk is split into  %% lines and each line is matched against each pattern.  %% one_expect: split data chunk at prompts @@ -1227,7 +1284,7 @@ one_expect(Name,Pid,Data,Pattern,EO) ->  		    log(name_or_pid(Name,Pid),"PROMPT: ~ts",[PromptType]),  		    {match,{prompt,PromptType},Rest};  		[{prompt,_OtherPromptType}] -> -		    %% Only searching for one specific prompt, not thisone +		    %% Only searching for one specific prompt, not this one  		    log_lines(Name,Pid,UptoPrompt),  		    {nomatch,Rest};  		_ -> diff --git a/lib/common_test/src/ct_telnet_client.erl b/lib/common_test/src/ct_telnet_client.erl index 0c448e5b35..757ccc0aae 100644 --- a/lib/common_test/src/ct_telnet_client.erl +++ b/lib/common_test/src/ct_telnet_client.erl @@ -39,7 +39,7 @@  -define(TELNET_PORT, 23).  -define(OPEN_TIMEOUT,10000). --define(IDLE_TIMEOUT,10000). +-define(IDLE_TIMEOUT,8000).  %% telnet control characters  -define(SE,	240). @@ -114,7 +114,7 @@ get_data(Pid) ->  %%%-----------------------------------------------------------------  %%% Internal functions  init(Parent, Server, Port, Timeout, KeepAlive, ConnName) -> -    case gen_tcp:connect(Server, Port, [list,{packet,0}], Timeout) of +    case gen_tcp:connect(Server, Port, [list,{packet,0},{nodelay,true}], Timeout) of  	{ok,Sock} ->  	    dbg("~p connected to: ~p (port: ~w, keep_alive: ~w)\n",  		[ConnName,Server,Port,KeepAlive]), @@ -393,7 +393,7 @@ cmd_dbg(Prefix,Cmd) ->      end.  timestamp() -> -    {MS,S,US} = now(), +    {MS,S,US} = os:timestamp(),      {{Year,Month,Day}, {Hour,Min,Sec}} =          calendar:now_to_local_time({MS,S,US}),      MilliSec = trunc(US/1000), diff --git a/lib/common_test/src/ct_testspec.erl b/lib/common_test/src/ct_testspec.erl index 10a9bdac67..10c3f2a938 100644 --- a/lib/common_test/src/ct_testspec.erl +++ b/lib/common_test/src/ct_testspec.erl @@ -27,6 +27,8 @@  	 collect_tests_from_list/2, collect_tests_from_list/3,  	 collect_tests_from_file/2, collect_tests_from_file/3]). +-export([testspec_rec2list/1, testspec_rec2list/2]). +  -include("ct_util.hrl").  -define(testspec_fields, record_info(fields, testspec)). @@ -973,7 +975,8 @@ add_tests([Term={Tag,all_nodes,Data}|Ts],Spec) ->  					should_be_added(Tag,Node,Data,Spec)],  	    add_tests(Tests++Ts,Spec);  	invalid ->				% ignore term -	    add_tests(Ts,Spec) +	    Unknown = Spec#testspec.unknown, +	    add_tests(Ts,Spec#testspec{unknown=Unknown++[Term]})      end;  %% create one test entry per node in Nodes and reinsert  add_tests([{Tag,[],Data}|Ts],Spec) -> @@ -1001,7 +1004,8 @@ add_tests([Term={Tag,NodeOrOther,Data}|Ts],Spec) ->  			handle_data(Tag,Node,Data,Spec),  		    add_tests(Ts,mod_field(Spec,Tag,NodeIxData));  		invalid ->			% ignore term -		    add_tests(Ts,Spec) +		    Unknown = Spec#testspec.unknown, +		    add_tests(Ts,Spec#testspec{unknown=Unknown++[Term]})  	    end;  	false ->  	    add_tests([{Tag,all_nodes,{NodeOrOther,Data}}|Ts],Spec) @@ -1012,13 +1016,15 @@ add_tests([Term={Tag,Data}|Ts],Spec) ->  	valid ->  	    add_tests([{Tag,all_nodes,Data}|Ts],Spec);  	invalid -> -	    add_tests(Ts,Spec) +	    Unknown = Spec#testspec.unknown, +	    add_tests(Ts,Spec#testspec{unknown=Unknown++[Term]})      end;  %% some other data than a tuple  add_tests([Other|Ts],Spec) ->	      case get(relaxed) of -	true ->		 -	    add_tests(Ts,Spec); +	true -> +	    Unknown = Spec#testspec.unknown, +	    add_tests(Ts,Spec#testspec{unknown=Unknown++[Other]});  	false ->  	    throw({error,{undefined_term_in_spec,Other}})      end; @@ -1149,6 +1155,24 @@ per_node([N|Ns],Tag,Data,Refs) ->  per_node([],_,_,_) ->      []. +%% Change the testspec record "back" to a list of tuples +testspec_rec2list(Rec) -> +    {Terms,_} = lists:mapfoldl(fun(unknown, Pos) -> +				       {element(Pos, Rec),Pos+1}; +				  (F, Pos) -> +				       {{F,element(Pos, Rec)},Pos+1} +			       end,2,?testspec_fields), +    lists:flatten(Terms). + +%% Extract one or more values from a testspec record and +%% return the result as a list of tuples +testspec_rec2list(Field, Rec) when is_atom(Field) -> +    [Term] = testspec_rec2list([Field], Rec), +    Term; +testspec_rec2list(Fields, Rec) -> +    Terms = testspec_rec2list(Rec), +    [{Field,proplists:get_value(Field, Terms)} || Field <- Fields]. +  %% read the value for FieldName in record Rec#testspec  read_field(Rec, FieldName) ->      catch lists:foldl(fun(F, Pos) when F == FieldName -> diff --git a/lib/common_test/src/ct_util.hrl b/lib/common_test/src/ct_util.hrl index 845bb55486..f4cf407856 100644 --- a/lib/common_test/src/ct_util.hrl +++ b/lib/common_test/src/ct_util.hrl @@ -55,6 +55,7 @@  		   create_priv_dir=[],  		   alias=[],  		   tests=[], +		   unknown=[],  		   merge_tests=true}).  -record(cover, {app=none, diff --git a/lib/common_test/src/ct_webtool.erl b/lib/common_test/src/ct_webtool.erl new file mode 100644 index 0000000000..b67a7c2a92 --- /dev/null +++ b/lib/common_test/src/ct_webtool.erl @@ -0,0 +1,1207 @@ +%% +%% %CopyrightBegin% +%% +%% Copyright Ericsson AB 2001-2010. 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(ct_webtool). +-behaviour(gen_server). + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%%                                                                    %% +%% The general idea is:                                               %% +%%                                                                    %% +%%                                                                    %% +%% 1. Scan through the path for *.tool files and find all the web     %% +%%    based tools. Query each tool for configuration data.            %% +%% 2. Add Alias for Erlscript and html for each tool to               %% +%%    the webserver configuration data.                               %% +%% 3. Start the webserver.                                            %% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% + +%% API functions +-export([start/0, start/2, stop/0]). + +%% Starting Webtool from a shell script +-export([script_start/0, script_start/1]). + +%% Web api +-export([started_tools/2, toolbar/2, start_tools/2, stop_tools/2]). + +%% API against other tools +-export([is_localhost/0]). + +%% Debug export s +-export([get_tools1/1]). +-export([debug/1, stop_debug/0, debug_app/1]). + +%% gen_server callbacks +-export([init/1, handle_call/3, handle_cast/2, handle_info/2, +	 terminate/2, code_change/3]). + +-include_lib("kernel/include/file.hrl"). +-include_lib("stdlib/include/ms_transform.hrl"). + +-record(state,{priv_dir,app_data,supvis,web_data,started=[]}). + +-define(MAX_NUMBER_OF_WEBTOOLS,256). +-define(DEFAULT_PORT,8888).% must be >1024 or the user must be root on unix +-define(DEFAULT_ADDR,{127,0,0,1}). + +-define(WEBTOOL_ALIAS,{ct_webtool,[{alias,{erl_alias,"/ct_webtool",[ct_webtool]}}]}). +-define(HEADER,"Pragma:no-cache\r\n Content-type: text/html\r\n\r\n"). +-define(HTML_HEADER,"<HTML>\r\n<HEAD>\r\n<TITLE>WebTool</TITLE>\r\n</HEAD>\r\n<BODY BGCOLOR=\"#FFFFFF\">\r\n"). +-define(HTML_HEADER_RELOAD,"<HTML>\r\n<HEAD>\r\n<TITLE>WebTool +                             </TITLE>\r\n</HEAD>\r\n +                             <BODY BGCOLOR=\"#FFFFFF\" onLoad=reloadCompiledList()>\r\n"). + +-define(HTML_END,"</BODY></HTML>"). + +-define(SEND_URL_TIMEOUT,5000). + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%%                                                                  %% +%% For debugging only.                                              %% +%%                                                                  %% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% Start tracing with +%% debug(Functions). +%% Functions = local | global | FunctionList +%% FunctionList = [Function] +%% Function = {FunctionName,Arity} | FunctionName | +%%            {Module, FunctionName, Arity} | {Module,FunctionName} +debug(F) ->  +    ttb:tracer(all,[{file,"webtool.trc"}]), % tracing all nodes +    ttb:p(all,[call,timestamp]), +    MS = [{'_',[],[{return_trace},{message,{caller}}]}], +    tp(F,MS), +    ttb:ctp(?MODULE,stop_debug), % don't want tracing of the stop_debug func +    ok. +tp(local,MS) -> % all functions +    ttb:tpl(?MODULE,MS); +tp(global,MS) -> % all exported functions +    ttb:tp(?MODULE,MS); +tp([{M,F,A}|T],MS) -> % Other module +    ttb:tpl(M,F,A,MS), +    tp(T,MS); +tp([{M,F}|T],MS) when is_atom(F) -> % Other module +    ttb:tpl(M,F,MS), +    tp(T,MS); +tp([{F,A}|T],MS) -> % function/arity +    ttb:tpl(?MODULE,F,A,MS), +    tp(T,MS); +tp([F|T],MS) -> % function +    ttb:tpl(?MODULE,F,MS), +    tp(T,MS); +tp([],_MS) -> +    ok. +stop_debug() -> +    ttb:stop([format]). + +debug_app(Mod) -> +    ttb:tracer(all,[{file,"webtool_app.trc"},{handler,{fun out/4,true}}]), +    ttb:p(all,[call,timestamp]), +    MS = [{'_',[],[{return_trace},{message,{caller}}]}], +    ttb:tp(Mod,MS), +    ok. +    +out(_,{trace_ts,Pid,call,MFA={M,F,A},{W,_,_},TS},_,S)  +  when W==webtool;W==mod_esi->  +    io:format("~w: (~p)~ncall ~s~n", [TS,Pid,ffunc(MFA)]), +    [{M,F,length(A)}|S]; +out(_,{trace_ts,Pid,return_from,MFA,R,TS},_,[MFA|S]) -> +    io:format("~w: (~p)~nreturned from ~s -> ~p~n", [TS,Pid,ffunc(MFA),R]), +    S; +out(_,_,_,_) -> +    ok. + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%%                                                                  %% +%% Functions called via script.                                     %% +%%                                                                  %% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +script_start() -> +    usage(), +    halt(). +script_start([App]) -> +    DefaultBrowser =  +	case os:type() of +	    {win32,_} -> iexplore; +	    _ -> firefox +	end, +    script_start([App,DefaultBrowser]); +script_start([App,Browser]) -> +    io:format("Starting webtool...\n"), +    start(), +    AvailableApps = get_applications(), +    {OSType,_} = os:type(), +    case lists:keysearch(App,1,AvailableApps) of +	{value,{App,StartPage}} -> +	    io:format("Starting ~w...\n",[App]), +	    start_tools([],"app=" ++ atom_to_list(App)), +	    PortStr = integer_to_list(get_port()), +	    Url = case StartPage of +		      "/" ++ Page ->  +			  "http://localhost:" ++ PortStr ++ "/" ++ Page; +		      _ ->  +			  "http://localhost:" ++ PortStr ++ "/" ++ StartPage +		  end, +	    case Browser of  +		none -> +		    ok; +                iexplore when OSType == win32-> +                    io:format("Starting internet explorer...\n"), +                    {ok,R} = win32reg:open(""), +		    Key="\\local_machine\\SOFTWARE\\Microsoft\\IE Setup\\Setup", +                    win32reg:change_key(R,Key), +                    {ok,Val} = win32reg:value(R,"Path"), +		    IExplore=filename:join(win32reg:expand(Val),"iexplore.exe"), +                    os:cmd("\"" ++ IExplore ++ "\" " ++ Url); +		_ when OSType == win32 -> +                    io:format("Starting ~w...\n",[Browser]), +                    os:cmd("\"" ++ atom_to_list(Browser) ++ "\" " ++ Url); +		B when B==firefox; B==mozilla -> +		    io:format("Sending URL to ~w...",[Browser]), +		    BStr = atom_to_list(Browser), +		    SendCmd = BStr ++ " -raise -remote \'openUrl(" ++  +			Url ++ ")\'", +		    Port = open_port({spawn,SendCmd},[exit_status]), +		    receive  +			{Port,{exit_status,0}} ->  +			    io:format("done\n"), +			    ok; +			{Port,{exit_status,_Error}} -> +			    io:format(" not running, starting ~w...\n", +				      [Browser]), +			    os:cmd(BStr ++ " " ++ Url), +			    ok +		    after ?SEND_URL_TIMEOUT -> +			    io:format(" failed, starting ~w...\n",[Browser]), +			    erlang:port_close(Port), +			    os:cmd(BStr ++ " " ++ Url) +		    end; +		_ -> +		    io:format("Starting ~w...\n",[Browser]), +		    os:cmd(atom_to_list(Browser) ++ " " ++ Url) +	    end, +	    ok; +	false -> +	    stop(), +	    io:format("\n{error,{unknown_app,~p}}\n",[App]), +	    halt() +    end. + +usage() -> +    io:format("Starting webtool...\n"), +    start(), +    Apps = lists:map(fun({A,_}) -> A end,get_applications()), +    io:format( +      "\nUsage: start_webtool application [ browser ]\n" +      "\nAvailable applications are: ~p\n" +      "Default browser is \'iexplore\' (Internet Explorer) on Windows " +      "or else \'firefox\'\n", +      [Apps]), +    stop(). + + +get_applications() -> +    gen_server:call(ct_web_tool,get_applications). +     +get_port() -> +    gen_server:call(ct_web_tool,get_port). + + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%%                                                                  %% +%% Api functions to the genserver.                                  %% +%%                                                                  %% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%---------------------------------------------------------------------- +% +%---------------------------------------------------------------------- + +start()-> +    start(standard_path,standard_data). + +start(Path,standard_data)-> +    case get_standard_data() of +	{error,Reason} -> +	    {error,Reason}; +	Data -> +	    start(Path,Data) +    end; +     +start(standard_path,Data)-> +    Path=get_path(), +    start(Path,Data); +     +start(Path,Port) when is_integer(Port)-> +    Data = get_standard_data(Port), +    start(Path,Data); +	 +start(Path,Data0)-> +    Data = Data0 ++ rest_of_standard_data(), +    gen_server:start({local,ct_web_tool},ct_webtool,{Path,Data},[]). + +stop()-> +    gen_server:call(ct_web_tool,stoppit). + +%---------------------------------------------------------------------- +%Web Api functions called by the web +%---------------------------------------------------------------------- +started_tools(Env,Input)-> +    gen_server:call(ct_web_tool,{started_tools,Env,Input}). + +toolbar(Env,Input)->     +    gen_server:call(ct_web_tool,{toolbar,Env,Input}). + +start_tools(Env,Input)-> +    gen_server:call(ct_web_tool,{start_tools,Env,Input}). + +stop_tools(Env,Input)-> +    gen_server:call(ct_web_tool,{stop_tools,Env,Input}). +%---------------------------------------------------------------------- +%Support API for other tools +%---------------------------------------------------------------------- + +is_localhost()-> +    gen_server:call(ct_web_tool,is_localhost). + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%%                                                                  %% +%%The gen_server callback functions that builds the webbpages       %% +%%                                                                  %% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +handle_call(get_applications,_,State)-> +    MS = ets:fun2ms(fun({Tool,{web_data,{_,Start}}}) -> {Tool,Start} end), +    Tools = ets:select(State#state.app_data,MS), +    {reply,Tools,State}; + +handle_call(get_port,_,State)-> +    {value,{port,Port}}=lists:keysearch(port,1,State#state.web_data), +    {reply,Port,State}; + +handle_call({started_tools,_Env,_Input},_,State)-> +    {reply,started_tools_page(State),State}; + +handle_call({toolbar,_Env,_Input},_,State)-> +    {reply,toolbar(),State}; + +handle_call({start_tools,Env,Input},_,State)-> +    {NewState,Page}=start_tools_page(Env,Input,State), +    {reply,Page,NewState}; + +handle_call({stop_tools,Env,Input},_,State)-> +    {NewState,Page}=stop_tools_page(Env,Input,State), +    {reply,Page,NewState}; + +handle_call(stoppit,_From,Data)-> +    {stop,normal,ok,Data}; + +handle_call(is_localhost,_From,Data)-> +    Result=case proplists:get_value(bind_address, Data#state.web_data) of +	?DEFAULT_ADDR -> +	    true; +	_IpNumber -> +	    false  +    end, +    {reply,Result,Data}. + + +handle_info(_Message,State)-> +    {noreply,State}. + +handle_cast(_Request,State)-> +    {noreply,State}. + +code_change(_,State,_)-> +    {ok,State}. +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +% +% The other functions needed by the gen_server behaviour  +% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%---------------------------------------------------------------------- +% Start the gen_server +%---------------------------------------------------------------------- +init({Path,Config})-> +    case filelib:is_dir(Path) of +	true -> +	    {ok, Table} = get_tool_files_data(), +	    insert_app(?WEBTOOL_ALIAS, Table), +	    case ct_webtool_sup:start_link() of +		{ok, Pid} -> +		    case start_webserver(Table, Path, Config) of +			{ok, _} -> +			    print_url(Config),	 +			    {ok,#state{priv_dir=Path, +				       app_data=Table, +				       supvis=Pid, +				       web_data=Config}}; +			{error, Error} -> +			    {stop, {error, Error}} +		    end; +		Error -> +		    {stop,Error} +	    end; +	false -> +	   {stop, {error, error_dir}} +    end. + +terminate(_Reason,Data)-> +    %%shut down the webbserver +    shutdown_server(Data), +    %%Shutdown the different tools that are started with application:start +    shutdown_apps(Data), +    %%Shutdown the supervisor and its children will die +    shutdown_supervisor(Data), +    ok. + +print_url(ConfigData)-> +    Server=proplists:get_value(server_name,ConfigData,"undefined"), +    Port=proplists:get_value(port,ConfigData,"undefined"), +    {A,B,C,D}=proplists:get_value(bind_address,ConfigData,"undefined"), +    io:format("WebTool is available at http://~s:~w/~n",[Server,Port]), +    io:format("Or  http://~w.~w.~w.~w:~w/~n",[A,B,C,D,Port]). + + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +% +% begin build the pages +% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% + +%---------------------------------------------------------------------- +%The page that shows the started tools +%---------------------------------------------------------------------- +started_tools_page(State)-> +    [?HEADER,?HTML_HEADER,started_tools(State),?HTML_END]. + +toolbar()-> +    [?HEADER,?HTML_HEADER,toolbar_page(),?HTML_END]. + +                +start_tools_page(_Env,Input,State)-> +    %%io:format("~n======= ~n ~p ~n============~n",[Input]), +    case get_tools(Input) of +	{tools,Tools}-> +	    %%io:format("~n======= ~n ~p ~n============~n",[Tools]), +	    {ok,NewState}=handle_apps(Tools,State,start), +	    {NewState,[?HEADER,?HTML_HEADER_RELOAD,reload_started_apps(), +		       show_unstarted_apps(NewState),?HTML_END]}; +	_ -> +	    {State,[?HEADER,?HTML_HEADER,show_unstarted_apps(State),?HTML_END]} +    end. + +stop_tools_page(_Env,Input,State)-> +    case get_tools(Input) of +	{tools,Tools}-> +	    {ok,NewState}=handle_apps(Tools,State,stop), +	    {NewState,[?HEADER,?HTML_HEADER_RELOAD,reload_started_apps(), +		       show_started_apps(NewState),?HTML_END]}; +	_ -> +	    {State,[?HEADER,?HTML_HEADER,show_started_apps(State),?HTML_END]} +    end. +	 + + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% +%% Functions that start and config the webserver +%% 1. Collect the config data +%% 2. Start webserver +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% + +%---------------------------------------------------------------------- +% Start the webserver +%---------------------------------------------------------------------- +start_webserver(Data,Path,Config)-> +    case get_conf_data(Data,Path,Config) of +	{ok,Conf_data}-> +	    %%io:format("Conf_data: ~p~n",[Conf_data]), +	    start_server(Conf_data); +	{error,Error} -> +	    {error,{error_server_conf_file,Error}} +    end. + +start_server(Conf_data)-> +    case inets:start(httpd, Conf_data, stand_alone) of +	{ok,Pid}-> +	    {ok,Pid}; +	Error-> +	    {error,{server_error,Error}} +    end. + +%---------------------------------------------------------------------- +% Create config data for the webserver  +%---------------------------------------------------------------------- +get_conf_data(Data,Path,Config)-> +    Aliases=get_aliases(Data), +    ServerRoot = filename:join([Path,"root"]), +    MimeTypesFile = filename:join([ServerRoot,"conf","mime.types"]), +    case httpd_conf:load_mime_types(MimeTypesFile) of +	{ok,MimeTypes} -> +	    Config1 = Config ++ Aliases, +	    Config2 = [{server_root,ServerRoot}, +			    {document_root,filename:join([Path,"root/doc"])}, +			    {mime_types,MimeTypes} | +			    Config1], +	    {ok,Config2}; +	Error -> +	    Error +    end. + +%---------------------------------------------------------------------- +% Control the path for *.tools files  +%---------------------------------------------------------------------- +get_tool_files_data()-> +    Tools=get_tools1(code:get_path()), +    %%io:format("Data : ~p ~n",[Tools]), +    get_file_content(Tools). + +%---------------------------------------------------------------------- +%Control that the data in the file really is erlang terms +%----------------------------------------------------------------------  +get_file_content(Tools)-> +    Get_data=fun({tool,ToolData}) -> +		     %%io:format("Data : ~p ~n",[ToolData]), +		     case proplists:get_value(config_func,ToolData) of +			 {M,F,A}-> +			     case catch apply(M,F,A) of +				 {'EXIT',_} -> +				     bad_data; +				 Data when is_tuple(Data) -> +				     Data; +				 _-> +				     bad_data   +			     end; +			 _ -> +				bad_data +		     end +	     end, +    insert_file_content([X ||X<-lists:map(Get_data,Tools),X/=bad_data]). + +%---------------------------------------------------------------------- +%Insert the data from the file in to the ets:table +%---------------------------------------------------------------------- +insert_file_content(Content)-> +    Table=ets:new(app_data,[bag]), +    lists:foreach(fun(X)-> +			  insert_app(X,Table) +		  end,Content), +    {ok,Table}. + +%---------------------------------------------------------------------- +%Control that we got a a tuple of a atom and a list if so add the  +%elements in the list to the ets:table +%---------------------------------------------------------------------- +insert_app({Name,Key_val_list},Table) when is_list(Key_val_list),is_atom(Name)-> +    %%io:format("ToolData: ~p: ~p~n",[Name,Key_val_list]), +    lists:foreach( +      fun({alias,{erl_alias,Alias,Mods}}) -> +	      Key_val = {erl_script_alias,{Alias,Mods}}, +	      %%io:format("Insert: ~p~n",[Key_val]), +	      ets:insert(Table,{Name,Key_val}); +	 (Key_val_pair)-> +	      %%io:format("Insert: ~p~n",[Key_val_pair]), +	      ets:insert(Table,{Name,Key_val_pair}) +      end, +      Key_val_list); + +insert_app(_,_)-> +    ok. +    +%---------------------------------------------------------------------- +% Select all the alias in the database +%---------------------------------------------------------------------- +get_aliases(Data)-> +    MS = ets:fun2ms(fun({_,{erl_script_alias,Alias}}) ->  +			    {erl_script_alias,Alias}; +		       ({_,{alias,Alias}}) ->  +			    {alias,Alias}  +		    end), +    ets:select(Data,MS). + + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%%                                                                  %% +%% Helper functions                                                 %% +%%                                                                  %% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +get_standard_data(Port)-> +    [ +     {port,Port}, +     {bind_address,?DEFAULT_ADDR}, +     {server_name,"localhost"} +    ]. + +get_standard_data()-> +    case get_free_port(?DEFAULT_PORT,?MAX_NUMBER_OF_WEBTOOLS) of +	{error,Reason} -> {error,Reason}; +	Port -> +	    [ +	     {port,Port}, +	     {bind_address,?DEFAULT_ADDR}, +	     {server_name,"localhost"} +	    ] +    end. + +get_free_port(_Port,0) -> +    {error,no_free_port_found}; +get_free_port(Port,N) -> +    case gen_tcp:connect("localhost",Port,[]) of +	{error, _Reason} -> +	    Port; +	{ok,Sock} -> +	    gen_tcp:close(Sock), +	    get_free_port(Port+1,N-1) +    end. + +rest_of_standard_data() -> +    [ +     %% Do not allow the server to be crashed by malformed http-request +     {max_header_siz,1024}, +     {max_header_action,reply414}, +     %% Go on a straight ip-socket +     {com_type,ip_comm}, +     %% Do not change the order of these module names!! +     {modules,[mod_alias, +	       mod_auth, +	       mod_esi, +	       mod_actions, +	       mod_cgi, +	       mod_include, +	       mod_dir, +	       mod_get, +	       mod_head, +	       mod_log, +	       mod_disk_log]}, +     {directory_index,["index.html"]}, +     {default_type,"text/plain"} +    ]. + + +get_path()-> +    code:priv_dir(webtool). + + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +% These functions is used to shutdown the webserver +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% + +%---------------------------------------------------------------------- +% Shut down the webbserver  +%---------------------------------------------------------------------- +shutdown_server(State)-> +    {Addr,Port} = get_addr_and_port(State#state.web_data), +    inets:stop(httpd,{Addr,Port}). + +get_addr_and_port(Config) -> +    Addr = proplists:get_value(bind_address,Config,?DEFAULT_ADDR), +    Port = proplists:get_value(port,Config,?DEFAULT_PORT), +    {Addr,Port}. + +%---------------------------------------------------------------------- +% Select all apps in the table and close them +%---------------------------------------------------------------------- +shutdown_apps(State)-> +    Data=State#state.app_data, +    MS = ets:fun2ms(fun({_,{start,HowToStart}}) -> HowToStart end), +    lists:foreach(fun(Start_app)-> +			  stop_app(Start_app) +		  end, +		  ets:select(Data,MS)). + +%---------------------------------------------------------------------- +%Shuts down the supervisor that supervises tools that is not +%Designed as applications +%---------------------------------------------------------------------- +shutdown_supervisor(State)-> +    %io:format("~n==================~n"), +    ct_webtool_sup:stop(State#state.supvis). +    %io:format("~n==================~n"). + +%---------------------------------------------------------------------- +%close the individual apps. +%----------------------------------------------------------------------   +stop_app({child,_Real_name})-> +    ok; + +stop_app({app,Real_name})-> +    application:stop(Real_name); + +stop_app({func,_Start,Stop})->     +    case Stop of +	{M,F,A} -> +	    catch apply(M,F,A); +	_NoStop -> +	    ok +    end. + + + + + + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% +%% These functions creates the webpage where the user can select if  +%% to start apps or to stop apps +%% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% + +toolbar_page()-> +    "<TABLE> +       <TR> +         <TD> +             <B>Select Action</B> +         </TD> +       </TR> +       <TR> +         <TD> +            <A HREF=\"./start_tools\" TARGET=right> Start Tools</A> +         </TD> +       </TR> +       <TR> +         <TD> +            <A HREF=\"./stop_tools\" TARGET=right> Stop Tools</A> +	 </TD>  +      </TR>  +    </TABLE>". +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% +%% These functions creates the webbpage that  shows the started apps +%% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% + +%---------------------------------------------------------------------- +% started_tools(State)->String (html table) +% State is a record of type state +%---------------------------------------------------------------------- +started_tools(State)-> +    Names=get_started_apps(State#state.app_data,State#state.started), +    "<TABLE BORDER=1 WIDTH=100%> +	"++ make_rows(Names,[],0) ++" +     </TABLE>". +%---------------------------------------------------------------------- +%get_started_apps(Data,Started)-> [{web_name,link}] +%selects the started apps from the ets table of apps. +%---------------------------------------------------------------------- +     +get_started_apps(Data,Started)-> +    SelectData=fun({Name,Link}) -> +		       {Name,Link} +	       end, +    MS = lists:map(fun(A) -> {{A,{web_data,'$1'}},[],['$1']} end,Started), + +    [{"WebTool","/tool_management.html"} |  +     [SelectData(X) || X <- ets:select(Data,MS)]]. + +%---------------------------------------------------------------------- +% make_rows(List,Result,Fields)-> String (The rows of a htmltable +% List a list of tupler discibed above +% Result an accumulator for the result +% Field, counter that counts the number of cols in each row. +%---------------------------------------------------------------------- +make_rows([],Result,Fields)-> +    Result ++ fill_out(Fields); +make_rows([Data|Paths],Result,Field)when Field==0-> +   make_rows(Paths,Result ++ "<TR>" ++ make_field(Data),Field+1); + +make_rows([Path|Paths],Result,Field)when Field==4-> +   make_rows(Paths,Result ++ make_field(Path) ++ "</TR>",0); + +make_rows([Path|Paths],Result,Field)-> +   make_rows(Paths,Result ++ make_field(Path),Field+1). + +%---------------------------------------------------------------------- +% make_fields(Path)-> String that is a field i a html table +% Path is a name url tuple {Name,url} +%---------------------------------------------------------------------- +make_field(Path)-> +    "<TD WIDTH=20%>" ++ get_name(Path) ++ "</TD>". + + +%---------------------------------------------------------------------- +%get_name({Nae,Url})->String that represents a <A> tag in html.  +%---------------------------------------------------------------------- +get_name({Name,Url})-> +    "<A HREF=\"" ++ Url ++ "\" TARGET=app_frame>" ++ Name ++ "</A>". + + +%---------------------------------------------------------------------- +% fill_out(Nr)-> String, that represent Nr fields in a html-table. +%---------------------------------------------------------------------- +fill_out(Nr)when Nr==0-> +    []; +fill_out(Nr)when Nr==4-> +    "<TD WIDTH=\"20%\" > </TD></TR>"; + +fill_out(Nr)-> +    "<TD WIDTH=\"20%\"> </TD>" ++ fill_out(Nr+1). + + + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% +%%These functions starts applicatons and builds the page showing tools +%%to start +%% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%----------------------------------------------------------------------  +%Controls whether the user selected a tool to start +%---------------------------------------------------------------------- +get_tools(Input)-> +    case httpd:parse_query(Input) of +	[]-> +	    no_tools; +	 Tools-> +	    FormatData=fun({_Name,Data}) -> list_to_atom(Data) end, +	    SelectData= +		fun({Name,_Data}) -> string:equal(Name,"app") end, +	    {tools,[FormatData(X)||X<-Tools,SelectData(X)]} +    end. + +%---------------------------------------------------------------------- +% Selects the data to start  the applications the user has ordered  +% starting of  +%---------------------------------------------------------------------- +handle_apps([],State,_Cmd)-> +    {ok,State}; + +handle_apps([Tool|Tools],State,Cmd)-> +    case ets:match_object(State#state.app_data,{Tool,{start,'_'}}) of +	[]-> +	    Started = case Cmd of +			  start -> +			      [Tool|State#state.started]; +			  stop -> +			      lists:delete(Tool,State#state.started) +		      end, +	    {ok,#state{priv_dir=State#state.priv_dir, +		       app_data=State#state.app_data, +		       supvis=State#state.supvis, +		       web_data=State#state.web_data, +		       started=Started}}; +	ToStart -> +	    case handle_apps2(ToStart,State,Cmd) of +		{ok,NewState}-> +		    handle_apps(Tools,NewState,Cmd); +		_-> +		    handle_apps(Tools,State,Cmd) +	    end +    end. + +%---------------------------------------------------------------------- +%execute every start or stop data about a tool. +%---------------------------------------------------------------------- +handle_apps2([{Name,Start_data}],State,Cmd)-> +    case handle_app({Name,Start_data},State#state.app_data,State#state.supvis,Cmd) of +	ok-> +	    Started = case Cmd of +			  start -> +			      [Name|State#state.started]; +			  stop -> +			       +			      lists:delete(Name,State#state.started) +		      end, +	    {ok,#state{priv_dir=State#state.priv_dir, +		       app_data=State#state.app_data, +		       supvis=State#state.supvis, +		       web_data=State#state.web_data, +		       started=Started}}; +	_-> +	    error +    end; + +handle_apps2([{Name,Start_data}|Rest],State,Cmd)-> +    case handle_app({Name,Start_data},State#state.app_data,State#state.supvis,Cmd)of +	ok-> +	    handle_apps2(Rest,State,Cmd); +	_-> +	    error +    end. + + +%---------------------------------------------------------------------- +% Handle start and stop of applications +%----------------------------------------------------------------------  + +handle_app({Name,{start,{func,Start,Stop}}},Data,_Pid,Cmd)-> +    Action = case Cmd of +		 start -> +		     Start; +		 _ -> +		     Stop +	     end,     +    case Action of +	{M,F,A} -> +	    case catch apply(M,F,A) of +		{'EXIT',_} = Exit-> +		    %%! Here the tool disappears from the webtool interface!! +		    io:format("\n=======ERROR (webtool, line ~w) =======\n" +			      "Could not start application \'~p\'\n\n" +			      "~w:~w(~s) ->\n" +			      "~p\n\n", +			      [?LINE,Name,M,F,format_args(A),Exit]), +		    ets:delete(Data,Name); +		_OK-> +		    ok +	    end; +	_NoStart -> +	    ok +    end; +	     + +handle_app({Name,{start,{child,ChildSpec}}},Data,Pid,Cmd)-> +    case Cmd of +	start -> +	    case catch supervisor:start_child(Pid,ChildSpec) of +		{ok,_}-> +		    ok; +		{ok,_,_}-> +		    ok; +		{error,Reason}-> +		    %%! Here the tool disappears from the webtool interface!! +		    io:format("\n=======ERROR (webtool, line ~w) =======\n" +			      "Could not start application \'~p\'\n\n" +			      "supervisor:start_child(~p,~p) ->\n" +			      "~p\n\n", +			      [?LINE,Name,Pid,ChildSpec,{error,Reason}]), +		    ets:delete(Data,Name); +		Error -> +		    %%! Here the tool disappears from the webtool interface!! +		    io:format("\n=======ERROR (webtool, line ~w) =======\n" +			      "Could not start application \'~p\'\n\n" +			      "supervisor:start_child(~p,~p) ->\n" +			      "~p\n\n", +			      [?LINE,Name,Pid,ChildSpec,Error]), +		    ets:delete(Data,Name) +	    end; +	stop -> +	    case catch supervisor:terminate_child(websup,element(1,ChildSpec)) of +		ok -> +		    supervisor:delete_child(websup,element(1,ChildSpec)); +		_ -> +		    error +	    end +    end; + + + +handle_app({Name,{start,{app,Real_name}}},Data,_Pid,Cmd)-> +    case Cmd of +	start -> +	    case application:start(Real_name,temporary) of +		ok-> +		    io:write(Name), +		    ok; +		{error,{already_started,_}}-> +		    %% Remove it from the database so we dont start +		    %% anything already started +		    ets:match_delete(Data,{Name,{start,{app,Real_name}}}), +		    ok; +		{error,_Reason}=Error-> +		    %%! Here the tool disappears from the webtool interface!! +		    io:format("\n=======ERROR (webtool, line ~w) =======\n" +			      "Could not start application \'~p\'\n\n" +			      "application:start(~p,~p) ->\n" +			      "~p\n\n", +			      [?LINE,Name,Real_name,temporary,Error]), +		    ets:delete(Data,Name) +	    end; +	 +	stop -> +	    application:stop(Real_name) +    end; + +%---------------------------------------------------------------------- +% If the data is incorrect delete the app +%---------------------------------------------------------------------- +handle_app({Name,Incorrect},Data,_Pid,Cmd)-> +    %%! Here the tool disappears from the webtool interface!! +    io:format("\n=======ERROR (webtool, line ~w) =======\n" +	      "Could not ~w application \'~p\'\n\n" +	      "Incorrect data: ~p\n\n", +	      [?LINE,Cmd,Name,Incorrect]), +    ets:delete(Data,Name). + + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%%                                                                  %% +%% this functions creates the page that shows the unstarted tools   %% +%%                                                                  %% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% + +reload_started_apps()-> +    "<script> +        function reloadCompiledList() +        { +          parent.parent.top1.document.location.href=\"/webtool/webtool/started_tools\"; +        } +     </script>". + +show_unstarted_apps(State)-> +  "<TABLE HEIGHT=100%  WIDTH=100% BORDER=0>  +    <TR HEIGHT=80%><TD ALIGN=\"center\" VALIGN=\"middle\">  +      <FORM NAME=\"stop_apps\" ACTION=\"/webtool/webtool/start_tools\" > +       <TABLE BORDER=1 WIDTH=60%> +	 <TR BGCOLOR=\"#8899AA\"> +	   <TD ALIGN=CENTER COLSPAN=2><FONT SIZE=4>Available Tools<FONT></TD> +	 </TR> + 	<TR> +	   <TD WIDTH=50%> +	       <TABLE BORDER=0> +	           "++ list_available_apps(State)++" +                   <TR><TD COLSPAN=2> </TD></TR> +                   <TR> +                      <TD COLSPAN=2 ALIGN=\"center\"> +                         <INPUT TYPE=submit VALUE=\"Start\"> +                      </TD> +                   </TR> +                </TABLE> +            </TD> +           <TD>    +             To Start a Tool: +             <UL> +             <LI>Select the +             checkbox for each tool to +             start.</LI> +             <LI>Click on the  +             button marked <EM>Start</EM>.</LI></UL> +            </TD>          +         </TR> +      </TABLE>  +    </FORM> +   </TD></TR> +   <TR><TD> </TD></TR> +   </TABLE>". + + + +list_available_apps(State)-> +    MS = ets:fun2ms(fun({Tool,{web_data,{Name,_}}}) -> {Tool,Name} end), +    Unstarted_apps= +	lists:filter( +	  fun({Tool,_})-> +		  false==lists:member(Tool,State#state.started) +	  end, +	  ets:select(State#state.app_data,MS)), +    case Unstarted_apps of +	[]-> +	    "<TR><TD>All tools are started</TD></TR>"; +	_-> +	    list_apps(Unstarted_apps) +    end. + + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%%                                                                  %% +%% these functions creates the page that shows the started apps     %% +%% the user can select to shutdown                                  %% +%%                                                                  %% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +show_started_apps(State)-> +  "<TABLE HEIGHT=100%  WIDTH=100% BORDER=0>  +    <TR HEIGHT=80%><TD ALIGN=\"center\" VALIGN=\"middle\">  +      <FORM NAME=\"stop_apps\" ACTION=\"/webtool/webtool/stop_tools\" > +       <TABLE BORDER=1 WIDTH=60%> +	 <TR BGCOLOR=\"#8899AA\"> +	   <TD ALIGN=CENTER COLSPAN=2><FONT SIZE=4>Started Tools<FONT></TD> +	 </TR> + 	<TR> +	   <TD WIDTH=50%> +	       <TABLE BORDER=0> +	           "++ list_started_apps(State)++" +                   <TR><TD COLSPAN=2> </TD></TR> +                   <TR> +                      <TD COLSPAN=2 ALIGN=\"center\"> +                         <INPUT TYPE=submit VALUE=\"Stop\"> +                      </TD> +                   </TR> +                </TABLE> +            </TD> +           <TD>    +             Stop a Tool: +             <UL> +             <LI>Select the +             checkbox for each tool to +             stop.</LI> +             <LI>Click on the  +             button marked <EM>Stop</EM>.</LI></UL> +            </TD>          +         </TR> +      </TABLE>  +    </FORM> +   </TD></TR> +   <TR><TD> </TD></TR> +   </TABLE>". + +list_started_apps(State)-> +    MS = lists:map(fun(A) -> {{A,{web_data,{'$1','_'}}},[],[{{A,'$1'}}]} end, +		   State#state.started), +    Started_apps= ets:select(State#state.app_data,MS), +    case Started_apps of +	[]-> +	    "<TR><TD>No tool is started yet.</TD></TR>"; +	_-> +	    list_apps(Started_apps) +    end. + + +list_apps(Apps) -> +      lists:map(fun({Tool,Name})-> +			"<TR><TD> +                            <INPUT TYPE=\"checkbox\" NAME=\"app\" VALUE=\""   +			    ++ atom_to_list(Tool) ++ "\"> +                               " ++ Name ++ "      +                            </TD></TR>" +		end, +		Apps). + + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%%                                                                   %% +%% Collecting the data from the  *.tool files                        %% +%%                                                                   %% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%---------------------------------------- +% get_tools(Dirs) => [{M,F,A},{M,F,A}...{M,F,A}] +%   Dirs - [string()] Directory names +% Calls get_tools2/2 recursively for a number of directories +% to retireve the configuration data for the web based tools. +%---------------------------------------- +get_tools1(Dirs)-> +    get_tools1(Dirs,[]). + +get_tools1([Dir|Rest],Data) when is_list(Dir) -> +    Tools=case filename:basename(Dir) of +	      %% Dir is an 'ebin' directory, check in '../priv' as well +	      "ebin" -> +		  [get_tools2(filename:join(filename:dirname(Dir),"priv")) | +		   get_tools2(Dir)]; +	      _ -> +		  get_tools2(Dir) +	  end, +    get_tools1(Rest,[Tools|Data]); + +get_tools1([],Data) -> +  lists:flatten(Data). + +%---------------------------------------- +% get_tools2(Directory) => DataList +%   DataList : [WebTuple]|[] +%   WebTuple: {tool,[{web,M,F,A}]} +% +%---------------------------------------- +get_tools2(Dir)-> +    get_tools2(tool_files(Dir),[]). + +get_tools2([ToolFile|Rest],Data) -> +    case get_tools3(ToolFile) of +	{tool,WebData} -> +	    get_tools2(Rest,[{tool,WebData}|Data]); +	{error,_Reason} -> +	    get_tools2(Rest,Data); +	nodata -> +	    get_tools2(Rest,Data) +    end; + +get_tools2([],Data) -> +    Data. + +%---------------------------------------- +% get_tools3(ToolFile) => {ok,Tool}|{error,Reason}|nodata  +%   Tool: {tool,[KeyValTuple]} +%   ToolFile - string() A .tool file +%   Now we have the file get the data and sort it out +%---------------------------------------- +get_tools3(ToolFile) -> +    case file:consult(ToolFile) of +	{error,open} -> +	    {error,nofile}; +	{error,read} -> +	    {error,format}; +	{ok,[{version,"1.2"},ToolInfo]} when is_list(ToolInfo)-> +	    webdata(ToolInfo); +	{ok,[{version,_Vsn},_Info]} -> +	    {error,old_version}; +	{ok,_Other} -> +	    {error,format} +    end. + + +%---------------------------------------------------------------------- +% webdata(TupleList)-> ToolTuple| nodata +% ToolTuple: {tool,[{config_func,{M,F,A}}]} +% +% There are a little unneccesary work in this format but it is extendable +%---------------------------------------------------------------------- +webdata(TupleList)->  +    case proplists:get_value(config_func,TupleList,nodata) of +	{M,F,A} -> +	    {tool,[{config_func,{M,F,A}}]}; +	_ -> +	   nodata +    end. + + +%============================================================================= +% Functions for getting *.tool configuration files +%============================================================================= + +%---------------------------------------- +% tool_files(Dir) => ToolFiles +%   Dir - string() Directory name +%   ToolFiles - [string()] +% Return the list of all files in Dir ending with .tool (appended to Dir) +%---------------------------------------- +tool_files(Dir) -> +    case file:list_dir(Dir) of +	{ok,Files} -> +	    filter_tool_files(Dir,Files); +	{error,_Reason} -> +	    [] +    end. + +%---------------------------------------- +% filter_tool_files(Dir,Files) => ToolFiles +%   Dir - string() Directory name +%   Files, ToolFiles - [string()] File names +% Filters out the files in Files ending with .tool and append them to Dir +%---------------------------------------- +filter_tool_files(_Dir,[]) -> +    []; +filter_tool_files(Dir,[File|Rest]) -> +    case filename:extension(File) of +	".tool" -> +	    [filename:join(Dir,File)|filter_tool_files(Dir,Rest)]; +	_ -> +	    filter_tool_files(Dir,Rest) +    end. + + +%%%----------------------------------------------------------------- +%%% format functions +ffunc({M,F,A}) when is_list(A) -> +    io_lib:format("~w:~w(~s)\n",[M,F,format_args(A)]); +ffunc({M,F,A}) when is_integer(A) -> +    io_lib:format("~w:~w/~w\n",[M,F,A]). + +format_args([]) -> +    ""; +format_args(Args) -> +    Str = lists:append(["~p"|lists:duplicate(length(Args)-1,",~p")]), +    io_lib:format(Str,Args). diff --git a/lib/common_test/src/ct_webtool_sup.erl b/lib/common_test/src/ct_webtool_sup.erl new file mode 100644 index 0000000000..1d612a2d18 --- /dev/null +++ b/lib/common_test/src/ct_webtool_sup.erl @@ -0,0 +1,74 @@ +%% +%% %CopyrightBegin% +%%  +%% Copyright Ericsson AB 2001-2009. 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(ct_webtool_sup). + +-behaviour(supervisor). + +%% External exports +-export([start_link/0,stop/1]). + +%% supervisor callbacks +-export([init/1]). + +%%%---------------------------------------------------------------------- +%%% API +%%%---------------------------------------------------------------------- +start_link() -> +    supervisor:start_link({local,ct_websup},ct_webtool_sup, []). + +stop(Pid)-> +   exit(Pid,normal). +%%%---------------------------------------------------------------------- +%%% Callback functions from supervisor +%%%---------------------------------------------------------------------- + +%%---------------------------------------------------------------------- +%% Func: init/1 +%% Returns: {ok,  {SupFlags,  [ChildSpec]}} | +%%          ignore                          | +%%          {error, Reason}    +%%---------------------------------------------------------------------- +init(_StartArgs) -> +    %%Child1 =  +    %%Child2 ={webcover_backend,{webcover_backend,start_link,[]},permanent,2000,worker,[webcover_backend]}, +    %%{ok,{{simple_one_for_one,5,10},[Child1]}}. +    {ok,{{one_for_one,100,10},[]}}. + +%%%---------------------------------------------------------------------- +%%% Internal functions +%%%---------------------------------------------------------------------- + + + + + + + + + + + + + + + + + + + diff --git a/lib/common_test/src/cth_surefire.erl b/lib/common_test/src/cth_surefire.erl index bb12171ea7..3deaefe0e9 100644 --- a/lib/common_test/src/cth_surefire.erl +++ b/lib/common_test/src/cth_surefire.erl @@ -59,6 +59,8 @@  -define(default_report,"junit_report.xml").  -define(suite_log,"suite.log.html"). +-define(now, os:timestamp()). +  %% Number of dirs from log root to testcase log file.  %% ct_run.<node>.<timestamp>/<test_name>/run.<timestamp>/<tc_log>.html  -define(log_depth,3). @@ -77,11 +79,11 @@ init(Path, Opts) ->  	    axis = proplists:get_value(axis,Opts,[]),  	    properties = proplists:get_value(properties,Opts,[]),  	    url_base = proplists:get_value(url_base,Opts), -	    timer = now() }. +	    timer = ?now }.  pre_init_per_suite(Suite,SkipOrFail,State) when is_tuple(SkipOrFail) ->      {SkipOrFail, init_tc(State#state{curr_suite = Suite, -				     curr_suite_ts = now()}, +				     curr_suite_ts = ?now},  			 SkipOrFail) };  pre_init_per_suite(Suite,Config,#state{ test_cases = [] } = State) ->      TcLog = proplists:get_value(tc_logfile,Config), @@ -96,7 +98,7 @@ pre_init_per_suite(Suite,Config,#state{ test_cases = [] } = State) ->  	end,      {Config, init_tc(State#state{ filepath = Path,  				  curr_suite = Suite, -				  curr_suite_ts = now(), +				  curr_suite_ts = ?now,  				  curr_log_dir = CurrLogDir},  		     Config) };  pre_init_per_suite(Suite,Config,State) -> @@ -169,9 +171,9 @@ do_tc_skip(Res, State) ->      State#state{ test_cases = [NewTC | tl(TCs)]}.  init_tc(State, Config) when is_list(Config) == false -> -    State#state{ timer = now(), tc_log =  "" }; +    State#state{ timer = ?now, tc_log =  "" };  init_tc(State, Config) -> -    State#state{ timer = now(), +    State#state{ timer = ?now,  		 tc_log =  proplists:get_value(tc_logfile, Config, [])}.  end_tc(Func, Config, Res, State) when is_atom(Func) -> @@ -194,7 +196,7 @@ end_tc(Name, _Config, _Res, State = #state{ curr_suite = Suite,      ClassName = atom_to_list(Suite),      PGroup = string:join([ atom_to_list(Group)||  			     Group <- lists:reverse(Groups)],"."), -    TimeTakes = io_lib:format("~f",[timer:now_diff(now(),TS) / 1000000]), +    TimeTakes = io_lib:format("~f",[timer:now_diff(?now,TS) / 1000000]),      State#state{ test_cases = [#testcase{ log = Log,  					  url = Url,  					  timestamp = now_to_string(TS), @@ -209,7 +211,7 @@ close_suite(#state{ test_cases = [] } = State) ->      State;  close_suite(#state{ test_cases = TCs, url_base = UrlBase } = State) ->      {Total,Fail,Skip} = count_tcs(TCs,0,0,0), -    TimeTaken = timer:now_diff(now(),State#state.curr_suite_ts) / 1000000, +    TimeTaken = timer:now_diff(?now,State#state.curr_suite_ts) / 1000000,      SuiteLog = filename:join(State#state.curr_log_dir,?suite_log),      SuiteUrl = make_url(UrlBase,SuiteLog),      Suite = #testsuite{ name = atom_to_list(State#state.curr_suite), diff --git a/lib/common_test/src/vts.erl b/lib/common_test/src/vts.erl index b340c6fdd1..ab13e7d0ee 100644 --- a/lib/common_test/src/vts.erl +++ b/lib/common_test/src/vts.erl @@ -63,21 +63,21 @@  %%%-----------------------------------------------------------------  %%% User API  start() -> -    webtool:start(), -    webtool:start_tools([],"app=vts"). +    ct_webtool:start(), +    ct_webtool:start_tools([],"app=vts").  init_data(ConfigFiles,EvHandlers,LogDir,LogOpts,Tests) ->      call({init_data,ConfigFiles,EvHandlers,LogDir,LogOpts,Tests}).  stop() -> -    webtool:stop_tools([],"app=vts"), -    webtool:stop(). +    ct_webtool:stop_tools([],"app=vts"), +    ct_webtool:stop().  report(What,Data) ->      call({report,What,Data}).  %%%----------------------------------------------------------------- -%%% Return config data used by webtool +%%% Return config data used by ct_webtool  config_data() ->      {ok,LogDir} =  	case lists:keysearch(logdir,1,init:get_arguments()) of diff --git a/lib/common_test/test/ct_auto_compile_SUITE.erl b/lib/common_test/test/ct_auto_compile_SUITE.erl index cc546ed30d..3e4da31ab4 100644 --- a/lib/common_test/test/ct_auto_compile_SUITE.erl +++ b/lib/common_test/test/ct_auto_compile_SUITE.erl @@ -108,6 +108,8 @@ ac_spec(Config) when is_list(Config) ->      PrivDir = ?config(priv_dir, Config),      file:copy(filename:join(DataDir, "bad_SUITE.erl"),  	      filename:join(PrivDir, "bad_SUITE.erl")), +    Suite = filename:join(DataDir, "dummy_SUITE"), +    compile:file(Suite, [{outdir,PrivDir}]),      TestSpec = [{label,ac_spec},  		{auto_compile,false},  		{suites,PrivDir,all}], @@ -160,28 +162,34 @@ events_to_check(Test, N) ->  test_events(ac_flag) ->      [ -     {ct_test_support_eh,start_logging,{'DEF','RUNDIR'}}, -     {ct_test_support_eh,test_start,{'DEF',{'START_TIME','LOGDIR'}}}, -     {ct_test_support_eh,start_info,{1,1,3}}, -     {ct_test_support_eh,tc_start,{dummy_SUITE,init_per_suite}}, -     {ct_test_support_eh,tc_done,{dummy_SUITE,init_per_suite,ok}}, -     {ct_test_support_eh,test_stats,{1,1,{1,0}}}, -     {ct_test_support_eh,tc_start,{dummy_SUITE,end_per_suite}}, -     {ct_test_support_eh,tc_done,{dummy_SUITE,end_per_suite,ok}}, -     {ct_test_support_eh,test_done,{'DEF','STOP_TIME'}}, -     {ct_test_support_eh,stop_logging,[]} +     {?eh,start_logging,{'DEF','RUNDIR'}}, +     {?eh,test_start,{'DEF',{'START_TIME','LOGDIR'}}}, +     {?eh,start_info,{1,1,3}}, +     {?eh,tc_start,{ct_framework,error_in_suite}}, +     {?eh,tc_done,{ct_framework,error_in_suite, +       {failed,{error,'bad_SUITE can not be compiled or loaded'}}}}, +     {?eh,tc_start,{dummy_SUITE,init_per_suite}}, +     {?eh,tc_done,{dummy_SUITE,init_per_suite,ok}}, +     {?eh,test_stats,{1,1,{1,0}}}, +     {?eh,tc_start,{dummy_SUITE,end_per_suite}}, +     {?eh,tc_done,{dummy_SUITE,end_per_suite,ok}}, +     {?eh,test_done,{'DEF','STOP_TIME'}}, +     {?eh,stop_logging,[]}      ];  test_events(ac_spec) ->      [ -     {ct_test_support_eh,start_logging,{'DEF','RUNDIR'}}, -     {ct_test_support_eh,test_start,{'DEF',{'START_TIME','LOGDIR'}}}, -     {ct_test_support_eh,start_info,{1,1,3}}, -     {ct_test_support_eh,tc_start,{dummy_SUITE,init_per_suite}}, -     {ct_test_support_eh,tc_done,{dummy_SUITE,init_per_suite,ok}}, -     {ct_test_support_eh,test_stats,{1,1,{1,0}}}, -     {ct_test_support_eh,tc_start,{dummy_SUITE,end_per_suite}}, -     {ct_test_support_eh,tc_done,{dummy_SUITE,end_per_suite,ok}}, -     {ct_test_support_eh,test_done,{'DEF','STOP_TIME'}}, -     {ct_test_support_eh,stop_logging,[]} +     {?eh,start_logging,{'DEF','RUNDIR'}}, +     {?eh,test_start,{'DEF',{'START_TIME','LOGDIR'}}}, +     {?eh,start_info,{1,1,3}}, +     {?eh,tc_start,{ct_framework,error_in_suite}}, +     {?eh,tc_done,{ct_framework,error_in_suite, +       {failed,{error,'bad_SUITE can not be compiled or loaded'}}}}, +     {?eh,tc_start,{dummy_SUITE,init_per_suite}}, +     {?eh,tc_done,{dummy_SUITE,init_per_suite,ok}}, +     {?eh,test_stats,{1,1,{1,0}}}, +     {?eh,tc_start,{dummy_SUITE,end_per_suite}}, +     {?eh,tc_done,{dummy_SUITE,end_per_suite,ok}}, +     {?eh,test_done,{'DEF','STOP_TIME'}}, +     {?eh,stop_logging,[]}      ]. diff --git a/lib/common_test/test/ct_config_SUITE_data/config/test/config_dynamic_SUITE.erl b/lib/common_test/test/ct_config_SUITE_data/config/test/config_dynamic_SUITE.erl index c2e06d866f..ef1fd63905 100644 --- a/lib/common_test/test/ct_config_SUITE_data/config/test/config_dynamic_SUITE.erl +++ b/lib/common_test/test/ct_config_SUITE_data/config/test/config_dynamic_SUITE.erl @@ -35,7 +35,7 @@  %% which will return the list with the following variables:  %% localtime = the erlang:localtime() result in list [{date, Date}, {time, Time}]  %% node = erlang:node() - can be compared in the testcase -%% now = erlang:now() - easier to compare than localtime() +%% now = os:timestamp() - easier to compare than localtime()  %% config_server_pid - pid of the config server, should NOT change!  %% config_server_vsn - .19  %% config_server_iteration - a number of iteration config_server's loop done diff --git a/lib/common_test/test/ct_config_SUITE_data/config/test/config_server.erl b/lib/common_test/test/ct_config_SUITE_data/config/test/config_server.erl index 8463fea645..e65d6584b1 100644 --- a/lib/common_test/test/ct_config_SUITE_data/config/test/config_server.erl +++ b/lib/common_test/test/ct_config_SUITE_data/config/test/config_server.erl @@ -73,7 +73,7 @@ loop(Iteration)->  		[{localtime, [{date, D}, {time, T}]},  		 {node, erlang:node()},  		 {config_server_iteration, Iteration}, -		 {now, erlang:now()}, +		 {now, os:timestamp()},  		 {config_server_pid, self()},  		 {config_server_vsn, ?vsn}],  	    Config2 = if Iteration rem 2 == 0-> diff --git a/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_2_SUITE.erl b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_2_SUITE.erl index a77d06815e..d926fc55a4 100644 --- a/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_2_SUITE.erl +++ b/lib/common_test/test/ct_error_SUITE_data/error/test/timetrap_2_SUITE.erl @@ -141,12 +141,13 @@ tc3() ->      [{timetrap,{seconds,2}}].  tc3(_) -> -    T0 = now(), +    T0 = erlang:monotonic_time(),      ct:timetrap(infinity),      N = list_to_integer(ct:get_config(multiply)),      ct:comment(io_lib:format("Sleeping for ~w sec...", [4*N])),      ct:sleep(4000), -    Diff = timer:now_diff(now(), T0), +    T1 = erlang:monotonic_time(), +    Diff = erlang:convert_time_unit(T1-T0, native, micro_seconds),      if ((Diff < (N*4000000)) or (Diff > (N*4500000))) ->  	    exit(not_expected);         true -> diff --git a/lib/common_test/test/ct_error_SUITE_data/error/test/verify_config.erl b/lib/common_test/test/ct_error_SUITE_data/error/test/verify_config.erl index 446dd8bfdf..d5b3e0035a 100644 --- a/lib/common_test/test/ct_error_SUITE_data/error/test/verify_config.erl +++ b/lib/common_test/test/ct_error_SUITE_data/error/test/verify_config.erl @@ -81,7 +81,7 @@ init(Id, Opts) ->  -spec id(Opts :: proplists:proplist()) ->      Id :: term().  id(Opts) -> -    now(). +    os:timestamp().  %% @doc Called before init_per_suite is called. Note that this callback is  %% only called if the CTH is added before init_per_suite is run (eg. in a test diff --git a/lib/common_test/test/ct_hooks_SUITE_data/cth/tests/ct_update_config_SUITE.erl b/lib/common_test/test/ct_hooks_SUITE_data/cth/tests/ct_update_config_SUITE.erl index 3c1f5669e8..f8c8725602 100644 --- a/lib/common_test/test/ct_hooks_SUITE_data/cth/tests/ct_update_config_SUITE.erl +++ b/lib/common_test/test/ct_hooks_SUITE_data/cth/tests/ct_update_config_SUITE.erl @@ -26,21 +26,23 @@  -include("ct.hrl").
 +-define(now, os:timestamp()).
 +
  %% Test server callback functions
  init_per_suite(Config) ->
 -    [{init_per_suite,now()}|Config].
 +    [{init_per_suite,?now}|Config].
  end_per_suite(_Config) ->
      ok.
  init_per_testcase(_TestCase, Config) ->
 -    [{init_per_testcase,now()}|Config].
 +    [{init_per_testcase,?now}|Config].
  end_per_testcase(_TestCase, _Config) ->
      ok.
  init_per_group(GroupName, Config) ->
 -    [{init_per_group,now()}|Config].
 +    [{init_per_group,?now}|Config].
  end_per_group(GroupName, Config) ->
      ok.
 diff --git a/lib/common_test/test/ct_hooks_SUITE_data/cth/tests/empty_cth.erl b/lib/common_test/test/ct_hooks_SUITE_data/cth/tests/empty_cth.erl index 77783fccf5..5f8eae1f70 100644 --- a/lib/common_test/test/ct_hooks_SUITE_data/cth/tests/empty_cth.erl +++ b/lib/common_test/test/ct_hooks_SUITE_data/cth/tests/empty_cth.erl @@ -87,7 +87,7 @@ id(Opts) ->      gen_event:notify(?CT_EVMGR_REF, #event{ name = cth, node = node(),  					    data = {?MODULE, id, [Opts]}}),      ct:log("~w:id called", [?MODULE]), -    now(). +    os:timestamp().  %% @doc Called before init_per_suite is called. Note that this callback is  %% only called if the CTH is added before init_per_suite is run (eg. in a test diff --git a/lib/common_test/test/ct_hooks_SUITE_data/cth/tests/update_config_cth.erl b/lib/common_test/test/ct_hooks_SUITE_data/cth/tests/update_config_cth.erl index 2ee0d7da9c..55a1b9a130 100644 --- a/lib/common_test/test/ct_hooks_SUITE_data/cth/tests/update_config_cth.erl +++ b/lib/common_test/test/ct_hooks_SUITE_data/cth/tests/update_config_cth.erl @@ -24,6 +24,7 @@  -include_lib("common_test/src/ct_util.hrl").
  -include_lib("common_test/include/ct_event.hrl").
 +-define(now, os:timestamp()).
  %% CT Hooks
  -compile(export_all).
 @@ -33,44 +34,44 @@ init(Id, Opts) ->  pre_init_per_suite(Suite, Config, State) ->
      empty_cth:pre_init_per_suite(Suite,Config,State),
 -    {[{pre_init_per_suite,now()}|Config],State}.
 +    {[{pre_init_per_suite,?now}|Config],State}.
  post_init_per_suite(Suite,Config,Return,State) ->
      empty_cth:post_init_per_suite(Suite,Config,Return,State),
 -    {[{post_init_per_suite,now()}|Return],State}.
 +    {[{post_init_per_suite,?now}|Return],State}.
  pre_end_per_suite(Suite,Config,State) ->
      empty_cth:pre_end_per_suite(Suite,Config,State),
 -    {[{pre_end_per_suite,now()}|Config],State}.
 +    {[{pre_end_per_suite,?now}|Config],State}.
  post_end_per_suite(Suite,Config,Return,State) ->
      empty_cth:post_end_per_suite(Suite,Config,Return,State),
 -    NewConfig = [{post_end_per_suite,now()}|Config],
 +    NewConfig = [{post_end_per_suite,?now}|Config],
      {NewConfig,NewConfig}.
  pre_init_per_group(Group,Config,State) ->
      empty_cth:pre_init_per_group(Group,Config,State),
 -    {[{pre_init_per_group,now()}|Config],State}.
 +    {[{pre_init_per_group,?now}|Config],State}.
  post_init_per_group(Group,Config,Return,State) ->
      empty_cth:post_init_per_group(Group,Config,Return,State),
 -    {[{post_init_per_group,now()}|Return],State}.
 +    {[{post_init_per_group,?now}|Return],State}.
  pre_end_per_group(Group,Config,State) ->
      empty_cth:pre_end_per_group(Group,Config,State),
 -    {[{pre_end_per_group,now()}|Config],State}.
 +    {[{pre_end_per_group,?now}|Config],State}.
  post_end_per_group(Group,Config,Return,State) ->
      empty_cth:post_end_per_group(Group,Config,Return,State),
 -    {[{post_end_per_group,now()}|Config],State}.
 +    {[{post_end_per_group,?now}|Config],State}.
  pre_init_per_testcase(TC,Config,State) ->
      empty_cth:pre_init_per_testcase(TC,Config,State),
 -    {[{pre_init_per_testcase,now()}|Config],State}.
 +    {[{pre_init_per_testcase,?now}|Config],State}.
  post_end_per_testcase(TC,Config,Return,State) ->
      empty_cth:post_end_per_testcase(TC,Config,Return,State),
 -    {[{post_end_per_testcase,now()}|Config],State}.
 +    {[{post_end_per_testcase,?now}|Config],State}.
  on_tc_fail(TC, Reason, State) ->
      empty_cth:on_tc_fail(TC,Reason,State).
 diff --git a/lib/common_test/test/ct_netconfc_SUITE.erl b/lib/common_test/test/ct_netconfc_SUITE.erl index c89a4cdabe..2959f77087 100644 --- a/lib/common_test/test/ct_netconfc_SUITE.erl +++ b/lib/common_test/test/ct_netconfc_SUITE.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2009-2012. All Rights Reserved. +%% Copyright Ericsson AB 2009-2014. 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 @@ -63,7 +63,8 @@ suite() -> [{ct_hooks,[ts_install_cth]}].  all() ->      [ -     default +     netconfc1_SUITE, +     netconfc_remote_SUITE      ].  %%-------------------------------------------------------------------- @@ -72,14 +73,21 @@ all() ->  %%%-----------------------------------------------------------------  %%% -default(Config) when is_list(Config) -> +netconfc1_SUITE(Config) when is_list(Config) ->      DataDir = ?config(data_dir, Config),      Suite = filename:join(DataDir, "netconfc1_SUITE"),      CfgFile = filename:join(DataDir, "netconfc1.cfg"),      {Opts,ERPid} = setup([{suite,Suite},{config,CfgFile}, -			  {label,default}], Config), +			  {label,netconfc1_SUITE}], Config), -    ok = execute(default, Opts, ERPid, Config). +    ok = execute(netconfc1_SUITE, Opts, ERPid, Config). + +netconfc_remote_SUITE(Config) when is_list(Config) -> +    DataDir = ?config(data_dir, Config), +    Suite = filename:join(DataDir, "netconfc_remote_SUITE"), +    {Opts,ERPid} = setup([{suite,Suite},{label,netconfc_remote_SUITE}], Config), + +    ok = execute(netconfc_remote_SUITE, Opts, ERPid, Config).  %%%----------------------------------------------------------------- @@ -112,16 +120,15 @@ reformat(Events, EH) ->  %%%-----------------------------------------------------------------  %%% TEST EVENTS  %%%----------------------------------------------------------------- -events_to_check(default,Config) -> -    {module,_} = code:load_abs(filename:join(?config(data_dir,Config), -					     netconfc1_SUITE)), -    TCs = netconfc1_SUITE:all(), -    code:purge(netconfc1_SUITE), -    code:delete(netconfc1_SUITE), +events_to_check(Suite,Config) -> +    {module,_} = code:load_abs(filename:join(?config(data_dir,Config),Suite)), +    TCs = Suite:all(), +    code:purge(Suite), +    code:delete(Suite),      OneTest =  	[{?eh,start_logging,{'DEF','RUNDIR'}}] ++ -	[{?eh,tc_done,{netconfc1_SUITE,TC,ok}} || TC <- TCs] ++ +	[{?eh,tc_done,{Suite,TC,ok}} || TC <- TCs] ++  	[{?eh,stop_logging,[]}],      %% 2 tests (ct:run_test + script_start) is default diff --git a/lib/common_test/test/ct_netconfc_SUITE_data/netconfc1_SUITE.erl b/lib/common_test/test/ct_netconfc_SUITE_data/netconfc1_SUITE.erl index a145d85b1d..e26ed4089a 100644 --- a/lib/common_test/test/ct_netconfc_SUITE_data/netconfc1_SUITE.erl +++ b/lib/common_test/test/ct_netconfc_SUITE_data/netconfc1_SUITE.erl @@ -1,7 +1,7 @@  %%--------------------------------------------------------------------  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2013. All Rights Reserved. +%% Copyright Ericsson AB 2013-2014. 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 @@ -30,25 +30,10 @@  -module(netconfc1_SUITE).  -include_lib("common_test/include/ct.hrl").  -include_lib("common_test/src/ct_netconfc.hrl"). --include_lib("public_key/include/public_key.hrl"). +-include("netconfc_test_lib.hrl").  -compile(export_all). -%% Default timetrap timeout (set in init_per_testcase). --define(default_timeout, ?t:minutes(1)). - --define(NS,ns). --define(LOCALHOST, "127.0.0.1"). --define(SSH_PORT, 2060). - --define(DEFAULT_SSH_OPTS,[{ssh,?LOCALHOST}, -			  {port,?SSH_PORT}, -			  {user,"xxx"}, -			  {password,"xxx"}]). --define(DEFAULT_SSH_OPTS(Dir), ?DEFAULT_SSH_OPTS++[{user_dir,Dir}]). - --define(ok,ok). -  suite() ->      [{ct_hooks, [{cth_conn_log,  		  [{ct_netconfc,[{log_type,html}, %will be overwritten by config @@ -91,6 +76,7 @@ all() ->  	     get_config,  	     get_config_xpath,  	     edit_config, +	     edit_config_opt_params,  	     copy_config,  	     delete_config,  	     lock, @@ -136,8 +122,8 @@ end_per_testcase(_Case, Config) ->  init_per_suite(Config) ->      case catch {crypto:start(), ssh:start()} of  	{ok, ok} -> -	    {ok, _} =  get_id_keys(Config), -	    make_dsa_files(Config), +	    {ok, _} =  netconfc_test_lib:get_id_keys(Config), +	    netconfc_test_lib:make_dsa_files(Config),  	    Server = ?NS:start(?config(data_dir,Config)),  	    [{server,Server}|Config];  	_ -> @@ -148,7 +134,7 @@ end_per_suite(Config) ->      ?NS:stop(?config(server,Config)),      ssh:stop(),      crypto:stop(), -    remove_id_keys(Config), +    netconfc_test_lib:remove_id_keys(Config),      Config.  hello(Config) -> @@ -164,7 +150,7 @@ hello_from_server_first(Config) ->      {ok,Client} = ct_netconfc:only_open(?DEFAULT_SSH_OPTS(DataDir)),      ct:sleep(500),      ?NS:expect(hello), -    ?ok = ct_netconfc:hello(Client), +    ?ok = ct_netconfc:hello(Client, [{capability, ["urn:com:ericsson:ebase:1.1.0"]}], infinity),      ?NS:expect_do_reply('close-session',close,ok),      ?ok = ct_netconfc:close_session(Client),      ok. @@ -415,6 +401,18 @@ edit_config(Config) ->      ?ok = ct_netconfc:close_session(Client),      ok. +edit_config_opt_params(Config) -> +    DataDir = ?config(data_dir,Config), +    {ok,Client} = open_success(DataDir), +    ?NS:expect_reply({'edit-config',{'default-operation',"none"}},ok), +    ?ok = ct_netconfc:edit_config(Client,running, +				  {server,[{xmlns,"myns"}], +				   [{name,["myserver"]}]}, +				  [{'default-operation',["none"]}]), +    ?NS:expect_do_reply('close-session',close,ok), +    ?ok = ct_netconfc:close_session(Client), +    ok. +  copy_config(Config) ->      DataDir = ?config(data_dir,Config),      {ok,Client} = open_success(DataDir), @@ -490,13 +488,16 @@ action(Config) ->      Data = [{myactionreturn,[{xmlns,"myns"}],["value"]}],      %% test either to receive {data,Data} or {ok,Data},      %% both need to be handled -    {Reply,RetVal} = case element(3, now()) rem 2 of -			 0 -> {{data,Data},{ok,Data}}; -			 1 -> {{ok,Data},ok} -		     end, -    ct:log("Client will receive {~w,Data}", [element(1,Reply)]), -    ?NS:expect_reply(action,Reply), -    RetVal = ct_netconfc:action(Client,{myaction,[{xmlns,"myns"}],[]}), +    ct:log("Client will receive {~w,~p}", [data,Data]), +    ct:log("Expecting ~p", [{ok, Data}]), +    ?NS:expect_reply(action,{data, Data}), +    {ok, Data} = ct_netconfc:action(Client,{myaction,[{xmlns,"myns"}],[]}), + +    ct:log("Client will receive {~w,~p}", [ok,Data]), +    ct:log("Expecting ~p", [ok]), +    ?NS:expect_reply(action,{ok, Data}), +    ok = ct_netconfc:action(Client,{myaction,[{xmlns,"myns"}],[]}), +      ?NS:expect_do_reply('close-session',close,ok),      ?ok = ct_netconfc:close_session(Client),      ok. @@ -1008,165 +1009,3 @@ pad(I) when I<10 ->      "0"++integer_to_list(I);  pad(I) ->      integer_to_list(I). - - -%%%----------------------------------------------------------------- -%%% BEGIN SSH key management -%% copy private keys to given dir from ~/.ssh -get_id_keys(Config) -> -    DstDir = ?config(priv_dir, Config), -    SrcDir = filename:join(os:getenv("HOME"), ".ssh"), -    RsaOk = copyfile(SrcDir, DstDir, "id_rsa"), -    DsaOk = copyfile(SrcDir, DstDir, "id_dsa"), -    case {RsaOk, DsaOk} of -	{{ok, _}, {ok, _}} -> {ok, both}; -	{{ok, _}, _} -> {ok, rsa}; -	{_, {ok, _}} -> {ok, dsa}; -	{Error, _} -> Error -    end. - -%% Remove later on. Use make_dsa_files instead. -remove_id_keys(Config) -> -    Dir = ?config(priv_dir, Config), -    file:delete(filename:join(Dir, "id_rsa")), -    file:delete(filename:join(Dir, "id_dsa")). - - -make_dsa_files(Config) -> -    make_dsa_files(Config, rfc4716_public_key). -make_dsa_files(Config, Type) -> -    {DSA, EncodedKey} = gen_dsa(128, 20), -    PKey = DSA#'DSAPrivateKey'.y, -    P = DSA#'DSAPrivateKey'.p, -    Q = DSA#'DSAPrivateKey'.q, -    G = DSA#'DSAPrivateKey'.g, -    Dss = #'Dss-Parms'{p=P, q=Q, g=G}, -    {ok, Hostname} = inet:gethostname(), -    {ok, {A, B, C, D}} = inet:getaddr(Hostname, inet), -    IP = lists:concat([A, ".", B, ".", C, ".", D]), -    Attributes = [], % Could be [{comment,"user@" ++ Hostname}], -    HostNames = [{hostnames,[IP, IP]}], -    PublicKey = [{{PKey, Dss}, Attributes}], -    KnownHosts = [{{PKey, Dss}, HostNames}], - -    KnownHostsEnc = public_key:ssh_encode(KnownHosts, known_hosts), -    KnownHosts = public_key:ssh_decode(KnownHostsEnc, known_hosts), - -    PublicKeyEnc = public_key:ssh_encode(PublicKey, Type), - -    SystemTmpDir = ?config(data_dir, Config), -    filelib:ensure_dir(SystemTmpDir), -    file:make_dir(SystemTmpDir), - -    DSAFile = filename:join(SystemTmpDir, "ssh_host_dsa_key.pub"), -    file:delete(DSAFile), - -    DSAPrivateFile  = filename:join(SystemTmpDir, "ssh_host_dsa_key"), -    file:delete(DSAPrivateFile), - -    KHFile = filename:join(SystemTmpDir, "known_hosts"), -    file:delete(KHFile), - -    PemBin = public_key:pem_encode([EncodedKey]), - -    file:write_file(DSAFile, PublicKeyEnc), -    file:write_file(KHFile, KnownHostsEnc), -    file:write_file(DSAPrivateFile, PemBin), -    ok. - - -%%-------------------------------------------------------------------- -%% @doc Creates a dsa key (OBS: for testing only) -%%   the sizes are in bytes -%% @spec (::integer()) -> {::atom(), ::binary(), ::opaque()} -%% @end -%%-------------------------------------------------------------------- -gen_dsa(LSize,NSize) when is_integer(LSize), is_integer(NSize) -> -    Key = gen_dsa2(LSize, NSize), -    {Key, encode_key(Key)}. - -encode_key(Key = #'DSAPrivateKey'{}) -> -    Der = public_key:der_encode('DSAPrivateKey', Key), -    {'DSAPrivateKey', Der, not_encrypted}. - -%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% -%% DSA key generation  (OBS: for testing only) -%% See http://en.wikipedia.org/wiki/Digital_Signature_Algorithm -%% and the fips_186-3.pdf -%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% -gen_dsa2(LSize, NSize) -> -    Q  = prime(NSize),  %% Choose N-bit prime Q -    X0 = prime(LSize), -    P0 = prime((LSize div 2) +1), -     -    %% Choose L-bit prime modulus P such that p-1 is a multiple of q. -    case dsa_search(X0 div (2*Q*P0), P0, Q, 1000) of -	error ->  -	    gen_dsa2(LSize, NSize); -	P ->	     -	    G = crypto:mod_pow(2, (P-1) div Q, P), % Choose G a number whose multiplicative order modulo p is q. -	    %%                 such that This may be done by setting g = h^(p-1)/q mod p, commonly h=2 is used. -	     -	    X = prime(20),               %% Choose x by some random method, where 0 < x < q. -	    Y = crypto:mod_pow(G, X, P), %% Calculate y = g^x mod p. -	     -	    #'DSAPrivateKey'{version=0, p = P, q = Q,  -			     g = crypto:bytes_to_integer(G), y = crypto:bytes_to_integer(Y), x = X} -    end. -     -%% See fips_186-3.pdf -dsa_search(T, P0, Q, Iter) when Iter > 0 -> -    P = 2*T*Q*P0 + 1, -    case is_prime(P, 50) of -	true -> P; -	false -> dsa_search(T+1, P0, Q, Iter-1) -    end; -dsa_search(_,_,_,_) ->  -    error. - - -%%%%%%% Crypto Math %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% -prime(ByteSize) -> -    Rand = odd_rand(ByteSize), -    prime_odd(Rand, 0). - -prime_odd(Rand, N) -> -    case is_prime(Rand, 50) of -	true ->  -	    Rand; -	false ->  -	    prime_odd(Rand+2, N+1) -    end. - -%% see http://en.wikipedia.org/wiki/Fermat_primality_test -is_prime(_, 0) -> true; -is_prime(Candidate, Test) ->  -    CoPrime = odd_rand(10000, Candidate), -    Result = crypto:mod_pow(CoPrime, Candidate, Candidate) , -    is_prime(CoPrime, crypto:bytes_to_integer(Result), Candidate, Test). - -is_prime(CoPrime, CoPrime, Candidate, Test) -> -    is_prime(Candidate, Test-1); -is_prime(_,_,_,_) -> -    false. - -odd_rand(Size) -> -    Min = 1 bsl (Size*8-1), -    Max = (1 bsl (Size*8))-1, -    odd_rand(Min, Max). - -odd_rand(Min,Max) -> -    Rand = crypto:rand_uniform(Min,Max), -    case Rand rem 2 of -	0 ->  -	    Rand + 1; -	_ ->  -	    Rand -    end. - -copyfile(SrcDir, DstDir, Fn) -> -    file:copy(filename:join(SrcDir, Fn), -	      filename:join(DstDir, Fn)). - -%%% END SSH key management -%%%----------------------------------------------------------------- diff --git a/lib/common_test/test/ct_netconfc_SUITE_data/netconfc_remote_SUITE.erl b/lib/common_test/test/ct_netconfc_SUITE_data/netconfc_remote_SUITE.erl new file mode 100644 index 0000000000..7a44d148dd --- /dev/null +++ b/lib/common_test/test/ct_netconfc_SUITE_data/netconfc_remote_SUITE.erl @@ -0,0 +1,147 @@ +%%-------------------------------------------------------------------- +%% %CopyrightBegin% +%% +%% Copyright Ericsson AB 2014. 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(netconfc_remote_SUITE). +-include_lib("common_test/include/ct.hrl"). +-include_lib("common_test/src/ct_netconfc.hrl"). +-include("netconfc_test_lib.hrl"). + +-compile(export_all). + +suite() -> +    [{ct_hooks, [{cth_conn_log,[{ct_netconfc,[{log_type,html}]}]}]}]. + +all() -> +    case os:find_executable("ssh") of +	false -> +	    {skip, "SSH not installed on host"}; +	_ -> +	    [remote_crash +	    ] +    end. + +groups() -> +    []. + +init_per_group(_GroupName, Config) -> +    Config. + +end_per_group(_GroupName, Config) -> +    Config. + +init_per_testcase(Case, Config) -> +    stop_node(Case), +    Dog = test_server:timetrap(?default_timeout), +    [{watchdog, Dog}|Config]. + +end_per_testcase(Case, Config) -> +    stop_node(Case), +    Dog=?config(watchdog, Config), +    test_server:timetrap_cancel(Dog), +    ok. + +stop_node(Case) -> +    {ok,Host} = inet:gethostname(), +    Node = list_to_atom("nc_" ++ atom_to_list(Case)++ "@" ++ Host), +    rpc:call(Node,erlang,halt,[]). + + +init_per_suite(Config) -> +    case {crypto:start(),ssh:start()} of +	{ok,ok} -> +	    {ok, _} =  netconfc_test_lib:get_id_keys(Config), +	    netconfc_test_lib:make_dsa_files(Config), +	    Config; +	_ -> +	    {skip, "Crypto and/or SSH could not be started locally!"} +    end. + +end_per_suite(Config) -> +    ssh:stop(), +    crypto:stop(), +    netconfc_test_lib:remove_id_keys(Config), +    Config. + +%% This test case is related to seq12645 +%% Running the netconf server in a remote node, test that the client +%% process terminates if the remote node goes down. +remote_crash(Config) -> +    {ok,Node} = ct_slave:start(nc_remote_crash), +    Pa = filename:dirname(code:which(?NS)), +    true = rpc:call(Node,code,add_patha,[Pa]), +     +    case {rpc:call(Node,crypto,start,[]),rpc:call(Node,ssh,start,[])} of +	{ok,ok} -> +	    Server = rpc:call(Node,?NS,start,[?config(data_dir,Config)]), +	    remote_crash(Node,Config); +	_ -> +	    {skip, "Crypto and/or SSH could not be started remote!"} +    end. + +remote_crash(Node,Config) -> +    DataDir = ?config(data_dir,Config), +    {ok,Client} = open_success(Node,DataDir), + +    ns(Node,expect_reply,[{'create-subscription',[stream]},ok]), +    ?ok = ct_netconfc:create_subscription(Client), + +    true = erlang:is_process_alive(Client), +    Ref = erlang:monitor(process,Client), +    rpc:call(Node,erlang,halt,[]), % take the node down as brutally as possible +    receive {'DOWN',Ref,process,Client,_} -> +	    ok +    after 10000 -> +	    ct:fail(client_still_alive) +    end. + +%%%----------------------------------------------------------------- + +break(_Config) -> +    test_server:break("break test case"). + +%%%----------------------------------------------------------------- +%% Open a netconf session which is not specified in a config file +open_success(Node,Dir) -> +    open_success(Node,Dir,[]). + +%% Open a netconf session which is not specified in a config file, and +%% give som extra options in addition to the test defaults. +open_success(Node,Dir,ExtraOpts) when is_list(Dir), is_list(ExtraOpts) -> +    ns(Node,hello,[1]), % tell server to send hello with session id 1 +    ns(Node,expect,[hello]), % tell server to expect a hello message from client +    open(Dir,ExtraOpts); + +%% Open a named netconf session which is not specified in a config file +open_success(Node,KeyOrName,Dir) when is_atom(KeyOrName), is_list(Dir) -> +    ns(Node,hello,[1]), +    ns(Node,expect,[hello]), +    ct_netconfc:open(KeyOrName,?DEFAULT_SSH_OPTS(Dir)). + +open(Dir) -> +    open(Dir,[]). +open(Dir,ExtraOpts) -> +    Opts = lists:ukeymerge(1,lists:keysort(1,ExtraOpts), +			   lists:keysort(1,?DEFAULT_SSH_OPTS(Dir))), +    ct_netconfc:open(Opts). + +%%%----------------------------------------------------------------- +%%% Call server on remote node +ns(Node,Func,Args) -> +    rpc:call(Node,?NS,Func,Args). + diff --git a/lib/common_test/test/ct_netconfc_SUITE_data/netconfc_test_lib.erl b/lib/common_test/test/ct_netconfc_SUITE_data/netconfc_test_lib.erl new file mode 100644 index 0000000000..e058bc7600 --- /dev/null +++ b/lib/common_test/test/ct_netconfc_SUITE_data/netconfc_test_lib.erl @@ -0,0 +1,166 @@ +-module(netconfc_test_lib). + +-export([get_id_keys/1, remove_id_keys/1, make_dsa_files/1]). +-include_lib("common_test/include/ct.hrl"). +-include_lib("public_key/include/public_key.hrl"). + +%%%----------------------------------------------------------------- +%%% BEGIN SSH key management +%% copy private keys to given dir from ~/.ssh +get_id_keys(Config) -> +    DstDir = ?config(priv_dir, Config), +    SrcDir = filename:join(os:getenv("HOME"), ".ssh"), +    RsaOk = copyfile(SrcDir, DstDir, "id_rsa"), +    DsaOk = copyfile(SrcDir, DstDir, "id_dsa"), +    case {RsaOk, DsaOk} of +	{{ok, _}, {ok, _}} -> {ok, both}; +	{{ok, _}, _} -> {ok, rsa}; +	{_, {ok, _}} -> {ok, dsa}; +	{Error, _} -> Error +    end. + +%% Remove later on. Use make_dsa_files instead. +remove_id_keys(Config) -> +    Dir = ?config(priv_dir, Config), +    file:delete(filename:join(Dir, "id_rsa")), +    file:delete(filename:join(Dir, "id_dsa")). + + +make_dsa_files(Config) -> +    make_dsa_files(Config, rfc4716_public_key). +make_dsa_files(Config, Type) -> +    {DSA, EncodedKey} = gen_dsa(128, 20), +    PKey = DSA#'DSAPrivateKey'.y, +    P = DSA#'DSAPrivateKey'.p, +    Q = DSA#'DSAPrivateKey'.q, +    G = DSA#'DSAPrivateKey'.g, +    Dss = #'Dss-Parms'{p=P, q=Q, g=G}, +    {ok, Hostname} = inet:gethostname(), +    {ok, {A, B, C, D}} = inet:getaddr(Hostname, inet), +    IP = lists:concat([A, ".", B, ".", C, ".", D]), +    Attributes = [], % Could be [{comment,"user@" ++ Hostname}], +    HostNames = [{hostnames,[IP, IP]}], +    PublicKey = [{{PKey, Dss}, Attributes}], +    KnownHosts = [{{PKey, Dss}, HostNames}], + +    KnownHostsEnc = public_key:ssh_encode(KnownHosts, known_hosts), +    KnownHosts = public_key:ssh_decode(KnownHostsEnc, known_hosts), + +    PublicKeyEnc = public_key:ssh_encode(PublicKey, Type), + +    SystemTmpDir = ?config(data_dir, Config), +    filelib:ensure_dir(SystemTmpDir), +    file:make_dir(SystemTmpDir), + +    DSAFile = filename:join(SystemTmpDir, "ssh_host_dsa_key.pub"), +    file:delete(DSAFile), + +    DSAPrivateFile  = filename:join(SystemTmpDir, "ssh_host_dsa_key"), +    file:delete(DSAPrivateFile), + +    KHFile = filename:join(SystemTmpDir, "known_hosts"), +    file:delete(KHFile), + +    PemBin = public_key:pem_encode([EncodedKey]), + +    file:write_file(DSAFile, PublicKeyEnc), +    file:write_file(KHFile, KnownHostsEnc), +    file:write_file(DSAPrivateFile, PemBin), +    ok. + + +%%-------------------------------------------------------------------- +%% @doc Creates a dsa key (OBS: for testing only) +%%   the sizes are in bytes +%% @spec (::integer()) -> {::atom(), ::binary(), ::opaque()} +%% @end +%%-------------------------------------------------------------------- +gen_dsa(LSize,NSize) when is_integer(LSize), is_integer(NSize) -> +    Key = gen_dsa2(LSize, NSize), +    {Key, encode_key(Key)}. + +encode_key(Key = #'DSAPrivateKey'{}) -> +    Der = public_key:der_encode('DSAPrivateKey', Key), +    {'DSAPrivateKey', Der, not_encrypted}. + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% DSA key generation  (OBS: for testing only) +%% See http://en.wikipedia.org/wiki/Digital_Signature_Algorithm +%% and the fips_186-3.pdf +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +gen_dsa2(LSize, NSize) -> +    Q  = prime(NSize),  %% Choose N-bit prime Q +    X0 = prime(LSize), +    P0 = prime((LSize div 2) +1), +     +    %% Choose L-bit prime modulus P such that p-1 is a multiple of q. +    case dsa_search(X0 div (2*Q*P0), P0, Q, 1000) of +	error ->  +	    gen_dsa2(LSize, NSize); +	P ->	     +	    G = crypto:mod_pow(2, (P-1) div Q, P), % Choose G a number whose multiplicative order modulo p is q. +	    %%                 such that This may be done by setting g = h^(p-1)/q mod p, commonly h=2 is used. +	     +	    X = prime(20),               %% Choose x by some random method, where 0 < x < q. +	    Y = crypto:mod_pow(G, X, P), %% Calculate y = g^x mod p. +	     +	    #'DSAPrivateKey'{version=0, p = P, q = Q,  +			     g = crypto:bytes_to_integer(G), y = crypto:bytes_to_integer(Y), x = X} +    end. +     +%% See fips_186-3.pdf +dsa_search(T, P0, Q, Iter) when Iter > 0 -> +    P = 2*T*Q*P0 + 1, +    case is_prime(P, 50) of +	true -> P; +	false -> dsa_search(T+1, P0, Q, Iter-1) +    end; +dsa_search(_,_,_,_) ->  +    error. + + +%%%%%%% Crypto Math %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +prime(ByteSize) -> +    Rand = odd_rand(ByteSize), +    prime_odd(Rand, 0). + +prime_odd(Rand, N) -> +    case is_prime(Rand, 50) of +	true ->  +	    Rand; +	false ->  +	    prime_odd(Rand+2, N+1) +    end. + +%% see http://en.wikipedia.org/wiki/Fermat_primality_test +is_prime(_, 0) -> true; +is_prime(Candidate, Test) ->  +    CoPrime = odd_rand(10000, Candidate), +    Result = crypto:mod_pow(CoPrime, Candidate, Candidate) , +    is_prime(CoPrime, crypto:bytes_to_integer(Result), Candidate, Test). + +is_prime(CoPrime, CoPrime, Candidate, Test) -> +    is_prime(Candidate, Test-1); +is_prime(_,_,_,_) -> +    false. + +odd_rand(Size) -> +    Min = 1 bsl (Size*8-1), +    Max = (1 bsl (Size*8))-1, +    odd_rand(Min, Max). + +odd_rand(Min,Max) -> +    Rand = crypto:rand_uniform(Min,Max), +    case Rand rem 2 of +	0 ->  +	    Rand + 1; +	_ ->  +	    Rand +    end. + +copyfile(SrcDir, DstDir, Fn) -> +    file:copy(filename:join(SrcDir, Fn), +	      filename:join(DstDir, Fn)). + +%%% END SSH key management +%%%----------------------------------------------------------------- diff --git a/lib/common_test/test/ct_netconfc_SUITE_data/netconfc_test_lib.hrl b/lib/common_test/test/ct_netconfc_SUITE_data/netconfc_test_lib.hrl new file mode 100644 index 0000000000..dcaad5ba93 --- /dev/null +++ b/lib/common_test/test/ct_netconfc_SUITE_data/netconfc_test_lib.hrl @@ -0,0 +1,14 @@ +%% Default timetrap timeout (set in init_per_testcase). +-define(default_timeout, ?t:minutes(1)). + +-define(NS,ns). % netconf server module +-define(LOCALHOST, "127.0.0.1"). +-define(SSH_PORT, 2060). + +-define(DEFAULT_SSH_OPTS,[{ssh,?LOCALHOST}, +			  {port,?SSH_PORT}, +			  {user,"xxx"}, +			  {password,"xxx"}]). +-define(DEFAULT_SSH_OPTS(Dir), ?DEFAULT_SSH_OPTS++[{user_dir,Dir}]). + +-define(ok,ok). diff --git a/lib/common_test/test/ct_netconfc_SUITE_data/ns.erl b/lib/common_test/test/ct_netconfc_SUITE_data/ns.erl index 27da67bd1d..e4bc396536 100644 --- a/lib/common_test/test/ct_netconfc_SUITE_data/ns.erl +++ b/lib/common_test/test/ct_netconfc_SUITE_data/ns.erl @@ -1,7 +1,7 @@  %%--------------------------------------------------------------------  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2012-2013. All Rights Reserved. +%% Copyright Ericsson AB 2012-2014. 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 @@ -382,6 +382,7 @@ event({startElement,_,Name,_,Attrs},[ignore,{se,Name,As}|Match]) ->  event({startPrefixMapping,_,Ns},[{ns,Ns}|Match]) -> Match;  event({startPrefixMapping,_,Ns},[ignore,{ns,Ns}|Match]) -> Match;  event({endPrefixMapping,_},Match) -> Match; +event({characters,Chs},[{characters,Chs}|Match]) -> Match;  event({endElement,_,Name,_},[{ee,Name}|Match]) -> Match;  event({endElement,_,Name,_},[ignore,{ee,Name}|Match]) -> Match;  event(endDocument,Match) when Match==[]; Match==[ignore] -> ok; @@ -471,14 +472,17 @@ capabilities(no_caps) ->  %%% expect_do_reply/3.  %%%  %%% match(term()) -> [Match]. -%%% Match = ignore | {se,Name} | {se,Name,Attrs} | {ee,Name} | {ns,Namespace} +%%% Match = ignore | {se,Name} | {se,Name,Attrs} | {ee,Name} | +%%%         {ns,Namespace} | {characters,Chs}  %%% Name = string() +%%% Chs = string()  %%% Attrs = [{atom(),string()}]  %%% Namespace = string()  %%%  %%% 'se' means start element, 'ee' means end element - i.e. to match  %%% an XML element you need one 'se' entry and one 'ee' entry with the -%%% same name in the match list. +%%% same name in the match list. 'characters' can be used for matching +%%% character data (cdata) inside an element.  match(hello) ->      [ignore,{se,"hello"},ignore,{ee,"hello"},ignore];  match('close-session') -> @@ -487,6 +491,10 @@ match('close-session') ->  match('edit-config') ->      [ignore,{se,"rpc"},{se,"edit-config"},{se,"target"},ignore,{ee,"target"},       {se,"config"},ignore,{ee,"config"},{ee,"edit-config"},{ee,"rpc"},ignore]; +match({'edit-config',{'default-operation',DO}}) -> +    [ignore,{se,"rpc"},{se,"edit-config"},{se,"target"},ignore,{ee,"target"}, +     {se,"default-operation"},{characters,DO},{ee,"default-operation"}, +     {se,"config"},ignore,{ee,"config"},{ee,"edit-config"},{ee,"rpc"},ignore];  match('get') ->      match({get,subtree});  match({'get',FilterType}) -> diff --git a/lib/common_test/test/ct_telnet_SUITE_data/ct_telnet_own_server_SUITE.erl b/lib/common_test/test/ct_telnet_SUITE_data/ct_telnet_own_server_SUITE.erl index 1d3f5918d2..9dc9095f47 100644 --- a/lib/common_test/test/ct_telnet_SUITE_data/ct_telnet_own_server_SUITE.erl +++ b/lib/common_test/test/ct_telnet_SUITE_data/ct_telnet_own_server_SUITE.erl @@ -40,6 +40,7 @@ all() ->       expect,       expect_repeat,       expect_sequence, +     expect_wait_until_prompt,       expect_error_prompt,       expect_error_timeout1,       expect_error_timeout2, @@ -81,6 +82,8 @@ end_per_group(_GroupName, Config) ->  expect(_) ->      {ok, Handle} = ct_telnet:open(telnet_server_conn1),      ok = ct_telnet:send(Handle, "echo ayt"), +    {ok,["ayt"]} = ct_telnet:expect(Handle, "ayt"), +    ok = ct_telnet:send(Handle, "echo ayt"),      {ok,["ayt"]} = ct_telnet:expect(Handle, ["ayt"]),      ok = ct_telnet:close(Handle),      ok. @@ -103,6 +106,21 @@ expect_sequence(_) ->      ok = ct_telnet:close(Handle),      ok. +%% Check that expect can wait for delayed prompt +expect_wait_until_prompt(_) -> +    {ok, Handle} = ct_telnet:open(telnet_server_conn1), +    Timeouts = [{idle_timeout,5000},{total_timeout,7000}], + +    ok = ct_telnet:send(Handle, "echo_delayed_prompt 3000 xxx"), +    {ok,["xxx"]} = +	ct_telnet:expect(Handle, "xxx", +			 [wait_for_prompt|Timeouts]), +    ok = ct_telnet:send(Handle, "echo_delayed_prompt 3000 yyy zzz"), +    {ok,[["yyy"],["zzz"]]} = +	ct_telnet:expect(Handle, ["yyy","zzz"], +			 [{wait_for_prompt,"> "}|Timeouts]), +    ok. +  %% Check that expect returns when a prompt is found, even if pattern  %% is not matched.  expect_error_prompt(_) -> diff --git a/lib/common_test/test/ct_test_server_if_1_SUITE.erl b/lib/common_test/test/ct_test_server_if_1_SUITE.erl index b6ef3062d4..214cb60c0d 100644 --- a/lib/common_test/test/ct_test_server_if_1_SUITE.erl +++ b/lib/common_test/test/ct_test_server_if_1_SUITE.erl @@ -236,14 +236,13 @@ test_events(ts_if_1) ->        {ts_if_2_SUITE,end_per_suite,         {failed,{error,{suite0_failed,{exited,suite0_goes_boom}}}}}}, -       {?eh,tc_start,{ct_framework,error_in_suite}}, -     {?eh,test_stats,{2,6,{4,7}}}, - +     {?eh,tc_done,{ct_framework,error_in_suite, +		   {failed,{error,'ts_if_3_SUITE:all/0 is missing'}}}},       {?eh,tc_start,{ct_framework,error_in_suite}}, -     {?eh,test_stats,{2,7,{4,7}}}, - +     {?eh,tc_done,{ct_framework,error_in_suite, +		   {failed,{error,'Bad return value from ts_if_4_SUITE:all/0'}}}},       {?eh,tc_start,{ts_if_5_SUITE,init_per_suite}},       {?eh,tc_done,{ts_if_5_SUITE,init_per_suite, @@ -252,7 +251,7 @@ test_events(ts_if_1) ->       {?eh,tc_auto_skip,        {ts_if_5_SUITE,my_test_case,         {require_failed_in_suite0,{not_available,undef_variable}}}}, -     {?eh,test_stats,{2,7,{4,8}}}, +     {?eh,test_stats,{2,5,{4,8}}},       {?eh,tc_auto_skip,        {ts_if_5_SUITE,end_per_suite,         {require_failed_in_suite0,{not_available,undef_variable}}}}, @@ -264,7 +263,7 @@ test_events(ts_if_1) ->       {?eh,tc_auto_skip,        {ts_if_6_SUITE,tc1,         {failed,{error,{suite0_failed,{exited,suite0_byebye}}}}}}, -     {?eh,test_stats,{2,7,{4,9}}}, +     {?eh,test_stats,{2,5,{4,9}}},       {?eh,tc_auto_skip,        {ct_framework,end_per_suite,         {failed,{error,{suite0_failed,{exited,suite0_byebye}}}}}}, @@ -274,13 +273,13 @@ test_events(ts_if_1) ->       {?eh,tc_done,{ct_framework,init_per_suite,ok}},       {?eh,tc_done,        {ts_if_7_SUITE,tc1,{auto_skipped,{testcase0_failed,bad_return_value}}}}, -     {?eh,test_stats,{2,7,{4,10}}}, +     {?eh,test_stats,{2,5,{4,10}}},       {?eh,tc_done,{ts_if_7_SUITE,  		   {init_per_group,g1,[]},  		   {auto_skipped,{group0_failed,bad_return_value}}}},       {?eh,tc_auto_skip,        {ts_if_7_SUITE,{tc2,g1},{group0_failed,bad_return_value}}}, -     {?eh,test_stats,{2,7,{4,11}}}, +     {?eh,test_stats,{2,5,{4,11}}},       {?eh,tc_auto_skip,        {ts_if_7_SUITE,{end_per_group,g1},{group0_failed,bad_return_value}}}, @@ -288,7 +287,7 @@ test_events(ts_if_1) ->        {?eh,tc_done,{ts_if_7_SUITE,{init_per_group,g2,[]},ok}},        {?eh,tc_done,{ts_if_7_SUITE,tc2,  		    {auto_skipped,{testcase0_failed,bad_return_value}}}}, -      {?eh,test_stats,{2,7,{4,12}}}, +      {?eh,test_stats,{2,5,{4,12}}},        {?eh,tc_start,{ts_if_7_SUITE,{end_per_group,g2,[]}}},        {?eh,tc_done,{ts_if_7_SUITE,{end_per_group,g2,[]},ok}}], @@ -300,17 +299,17 @@ test_events(ts_if_1) ->       {?eh,tc_done,{ct_framework,init_per_suite,ok}},       {?eh,tc_start,{ts_if_8_SUITE,tc1}},       {?eh,tc_done,{ts_if_8_SUITE,tc1,{failed,{error,failed_on_purpose}}}}, -     {?eh,test_stats,{2,8,{4,12}}}, +     {?eh,test_stats,{2,6,{4,12}}},       {?eh,tc_start,{ct_framework,end_per_suite}},       {?eh,tc_done,{ct_framework,end_per_suite,ok}},       {?eh,tc_user_skip,{skipped_by_spec_1_SUITE,all,"should be skipped"}}, -     {?eh,test_stats,{2,8,{5,12}}}, +     {?eh,test_stats,{2,6,{5,12}}},       {?eh,tc_start,{skipped_by_spec_2_SUITE,init_per_suite}},       {?eh,tc_done,{skipped_by_spec_2_SUITE,init_per_suite,ok}},       {?eh,tc_user_skip,{skipped_by_spec_2_SUITE,tc1,"should be skipped"}}, -     {?eh,test_stats,{2,8,{6,12}}}, +     {?eh,test_stats,{2,6,{6,12}}},       {?eh,tc_start,{skipped_by_spec_2_SUITE,end_per_suite}},       {?eh,tc_done,{skipped_by_spec_2_SUITE,end_per_suite,ok}}, diff --git a/lib/common_test/test/ct_test_support.erl b/lib/common_test/test/ct_test_support.erl index 6abca08452..ffef8187f3 100644 --- a/lib/common_test/test/ct_test_support.erl +++ b/lib/common_test/test/ct_test_support.erl @@ -287,10 +287,13 @@ run_ct_run_test(Opts,Config) ->      Level = proplists:get_value(trace_level, Config),      test_server:format(Level, "~n[RUN #1] Calling ct:run_test(~p) on ~p~n",  		       [Opts, CTNode]), -    T0 = now(), +     +    T0 = erlang:monotonic_time(),      CtRunTestResult = rpc:call(CTNode, ct, run_test, [Opts]), +    T1 = erlang:monotonic_time(), +    Elapsed = erlang:convert_time_unit(T1-T0, native, milli_seconds),      test_server:format(Level, "~n[RUN #1] Got return value ~p after ~p ms~n", -		       [CtRunTestResult,trunc(timer:now_diff(now(), T0)/1000)]), +		       [CtRunTestResult,Elapsed]),      case rpc:call(CTNode, erlang, whereis, [ct_util_server]) of  	undefined ->  	    ok; @@ -313,10 +316,12 @@ run_ct_script_start(Opts, Config) ->  	     [common_test, run_test_start_opts, Opts1]),      test_server:format(Level, "[RUN #2] Calling ct_run:script_start() on ~p~n",  		       [CTNode]), -    T0 = now(), +    T0 = erlang:monotonic_time(),      ExitStatus = rpc:call(CTNode, ct_run, script_start, []), +    T1 = erlang:monotonic_time(), +    Elapsed = erlang:convert_time_unit(T1-T0, native, milli_seconds),      test_server:format(Level, "[RUN #2] Got exit status value ~p after ~p ms~n", -		       [ExitStatus,trunc(timer:now_diff(now(), T0)/1000)]), +		       [ExitStatus,Elapsed]),      ExitStatus.  check_result({_Ok,Failed,{_UserSkipped,_AutoSkipped}},1,_Opts) @@ -408,7 +413,7 @@ ct_rpc({M,F,A}, Config) ->  %%%-----------------------------------------------------------------  %%% random_error/1  random_error(Config) when is_list(Config) -> -    random:seed(now()), +    random:seed(os:timestamp()),      Gen = fun(0,_) -> ok; (N,Fun) -> Fun(N-1, Fun) end,      Gen(random:uniform(100), Gen), @@ -1350,12 +1355,7 @@ delete_old_logs(_, Config) ->  delete_dirs(LogDir) ->      Now = calendar:datetime_to_gregorian_seconds(calendar:local_time()), -    SaveTime = case os:getenv("CT_SAVE_OLD_LOGS") of -		   false -> -		       28800; -		   SaveTime0 -> -		       list_to_integer(SaveTime0) -	       end, +    SaveTime = list_to_integer(os:getenv("CT_SAVE_OLD_LOGS", "28800")),      Deadline = Now - SaveTime,      Dirs = filelib:wildcard(filename:join(LogDir,"ct_run*")),      Dirs2Del = diff --git a/lib/common_test/test/ct_testspec_3_SUITE_data/tests1/t11_SUITE.erl b/lib/common_test/test/ct_testspec_3_SUITE_data/tests1/t11_SUITE.erl index b8216c3596..cfc6fa93d7 100644 --- a/lib/common_test/test/ct_testspec_3_SUITE_data/tests1/t11_SUITE.erl +++ b/lib/common_test/test/ct_testspec_3_SUITE_data/tests1/t11_SUITE.erl @@ -41,8 +41,12 @@ suite() ->  %% @end  %%--------------------------------------------------------------------  init_per_suite(Config) -> + +    TCName = ct:get_config(tcname), +    CfgFiles = ct:get_config(file,undefined,[all]), +      %% verify that expected config file can be read -    case {ct:get_config(tcname),ct:get_config(file,undefined,[all])} of +    case {TCName,CfgFiles} of  	{start_separate,[cfg11]} -> ok;  	{start_join,[cfg11,cfg21]} -> ok;  	{incl_separate1,[cfg11]} -> ok; @@ -56,6 +60,28 @@ init_per_suite(Config) ->  	_ -> ok      end, + +    %% test the get_testspec_terms functionality +    if CfgFiles /= undefined -> +	    TSTerms = case ct:get_testspec_terms() of +			  undefined -> exit('testspec should not be undefined'); +			  Result -> Result +		      end, +	    true = lists:keymember(config, 1, TSTerms), +	    {config,TSCfgFiles} = ct:get_testspec_terms(config), +	    [{config,TSCfgFiles},{tests,Tests}] =  +		ct:get_testspec_terms([config,tests]), +	    CfgNames = [list_to_atom(filename:basename(TSCfgFile)) || +			   {Node,TSCfgFile} <- TSCfgFiles, Node == node()], +	    true = (length(CfgNames) == length(CfgFiles)), +	    [true = lists:member(CfgName,CfgFiles) || CfgName <- CfgNames], +	    true = lists:any(fun({{_Node,_Dir},Suites}) -> +				     lists:keymember(?MODULE, 1, Suites) +			     end, Tests); +       true -> +	    ok +    end, +      Config.  %%-------------------------------------------------------------------- diff --git a/lib/common_test/test/ct_testspec_3_SUITE_data/tests1/t12_SUITE.erl b/lib/common_test/test/ct_testspec_3_SUITE_data/tests1/t12_SUITE.erl index 7c51aca246..c3faebbd64 100644 --- a/lib/common_test/test/ct_testspec_3_SUITE_data/tests1/t12_SUITE.erl +++ b/lib/common_test/test/ct_testspec_3_SUITE_data/tests1/t12_SUITE.erl @@ -55,7 +55,7 @@ init_per_suite(Config) ->  	{incl_both2,[cfg11,cfg12,cfg21]} -> ok;  	{incl_both2,[cfg21]} -> ok;  	_ -> ok -    end, +    end,       Config.  %%-------------------------------------------------------------------- diff --git a/lib/common_test/test/ct_testspec_3_SUITE_data/tests2/t21_SUITE.erl b/lib/common_test/test/ct_testspec_3_SUITE_data/tests2/t21_SUITE.erl index 36c1b4279b..e189b168c7 100644 --- a/lib/common_test/test/ct_testspec_3_SUITE_data/tests2/t21_SUITE.erl +++ b/lib/common_test/test/ct_testspec_3_SUITE_data/tests2/t21_SUITE.erl @@ -41,8 +41,11 @@ suite() ->  %% @end  %%--------------------------------------------------------------------  init_per_suite(Config) -> +    TCName = ct:get_config(tcname), +    CfgFiles = ct:get_config(file,undefined,[all]), +      %% verify that expected config file can be read -    case {ct:get_config(tcname),ct:get_config(file,undefined,[all])} of +    case {TCName,CfgFiles} of   	{start_separate,[cfg11]} -> ok;  	{start_join,[cfg11,cfg21]} -> ok;  	{incl_separate1,[cfg11]} -> ok; @@ -55,6 +58,28 @@ init_per_suite(Config) ->  	{incl_both2,[cfg11]} -> ok;  	_ -> ok      end, + +     %% test the get_testspec_terms functionality +    if CfgFiles /= undefined -> +	    TSTerms = case ct:get_testspec_terms() of +			  undefined -> exit('testspec should not be undefined'); +			  Result -> Result +		      end, +	    true = lists:keymember(config, 1, TSTerms), +	    {config,TSCfgFiles} = ct:get_testspec_terms(config), +	    [{config,TSCfgFiles},{tests,Tests}] =  +		ct:get_testspec_terms([config,tests]), +	    CfgNames = [list_to_atom(filename:basename(TSCfgFile)) || +			   {Node,TSCfgFile} <- TSCfgFiles, Node == node()], +	    true = (length(CfgNames) == length(CfgFiles)), +	    [true = lists:member(CfgName,CfgFiles) || CfgName <- CfgNames], +	    true = lists:any(fun({{_Node,_Dir},Suites}) -> +				     lists:keymember(?MODULE, 1, Suites) +			     end, Tests); +       true -> +	    ok +    end, +      Config.  %%-------------------------------------------------------------------- diff --git a/lib/common_test/test/telnet_server.erl b/lib/common_test/test/telnet_server.erl index 11959c3e12..107d98d72c 100644 --- a/lib/common_test/test/telnet_server.erl +++ b/lib/common_test/test/telnet_server.erl @@ -242,6 +242,12 @@ do_handle_data("echo_loop " ++ Data,State) ->      ReturnData = string:join(Lines,"\n"),      send_loop(list_to_integer(TStr),ReturnData,State),      {ok,State}; +do_handle_data("echo_delayed_prompt "++Data,State) -> +    [MsStr|EchoData] = string:tokens(Data, " "), +    send(string:join(EchoData,"\n"),State), +    ct:sleep(list_to_integer(MsStr)), +    send("\r\n> ",State), +    {ok,State};  do_handle_data("disconnect_after " ++WaitStr,State) ->      Wait = list_to_integer(string:strip(WaitStr,right,$\n)),      dbg("Server will close connection in ~w ms...", [Wait]), @@ -284,10 +290,10 @@ send(Data,State) ->  send_loop(T,Data,State) ->      dbg("Server sending ~p in loop for ~w ms...~n",[Data,T]), -    send_loop(now(),T,Data,State). +    send_loop(os:timestamp(),T,Data,State).  send_loop(T0,T,Data,State) -> -    ElapsedMS = trunc(timer:now_diff(now(),T0)/1000), +    ElapsedMS = trunc(timer:now_diff(os:timestamp(),T0)/1000),      if ElapsedMS >= T ->  	    ok;         true -> @@ -314,7 +320,7 @@ dbg(_F,_A) ->      io:format("[telnet_server, ~s]\n" ++ _F,[TS|_A]).  timestamp() -> -    {MS,S,US} = now(), +    {MS,S,US} = os:timestamp(),      {{Year,Month,Day}, {Hour,Min,Sec}} =          calendar:now_to_local_time({MS,S,US}),      MilliSec = trunc(US/1000), diff --git a/lib/common_test/vsn.mk b/lib/common_test/vsn.mk index d654a8afb3..ff2bd20ab3 100644 --- a/lib/common_test/vsn.mk +++ b/lib/common_test/vsn.mk @@ -1 +1 @@ -COMMON_TEST_VSN = 1.10 +COMMON_TEST_VSN = 1.11 | 
