diff options
Diffstat (limited to 'lib/common_test/src')
39 files changed, 1048 insertions, 1014 deletions
| diff --git a/lib/common_test/src/common_test.app.src b/lib/common_test/src/common_test.app.src index 430a4fa2fb..0aa4aacf16 100644 --- a/lib/common_test/src/common_test.app.src +++ b/lib/common_test/src/common_test.app.src @@ -92,7 +92,7 @@      "sasl-2.4.2",      "snmp-5.1.2",      "ssh-4.0", -    "stdlib-2.5", +    "stdlib-3.4",      "syntax_tools-1.7",      "tools-2.8",      "xmerl-1.3.8" diff --git a/lib/common_test/src/ct.erl b/lib/common_test/src/ct.erl index 43abb91819..a12c0c9101 100644 --- a/lib/common_test/src/ct.erl +++ b/lib/common_test/src/ct.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2003-2016. All Rights Reserved. +%% Copyright Ericsson AB 2003-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -89,6 +89,36 @@  -export([get_target_name/1]).  -export([parse_table/1, listenv/1]). +%%---------------------------------------------------------------------- +%% Exported types +%%---------------------------------------------------------------------- +%% For ct_gen_conn +-export_type([config_key/0, +	      target_name/0, +	      key_or_name/0]). + +%% For cth_conn_log +-export_type([conn_log_options/0, +	      conn_log_type/0, +	      conn_log_mod/0]). + +%%------------------------------------------------------------------ +%% Type declarations +%% ------------------------------------------------------------------ +-type config_key() :: atom(). % Config key which exists in a config file +-type target_name() :: atom().% Name associated to a config_key() though 'require' +-type key_or_name() :: config_key() | target_name(). + +%% Types used when logging connections with the 'cth_conn_log' hook +-type conn_log_options() :: [conn_log_option()]. +-type conn_log_option() :: {log_type,conn_log_type()} | +                           {hosts,[key_or_name()]}. +-type conn_log_type() :: raw | pretty | html | silent. +-type conn_log_mod() :: ct_netconfc | ct_telnet. +%%---------------------------------------------------------------------- + + +  %%%-----------------------------------------------------------------  %%% @spec install(Opts) -> ok | {error,Reason}  %%%    Opts = [Opt] @@ -818,7 +848,8 @@ capture_get([ExclCat | ExclCategories]) ->      Strs = test_server:capture_get(),      CatsStr = [atom_to_list(ExclCat) |   	       [[$| | atom_to_list(EC)] || EC <- ExclCategories]], -    {ok,MP} = re:compile("<div class=\"(" ++ lists:flatten(CatsStr) ++ ")\">.*"), +    {ok,MP} = re:compile("<div class=\"(" ++ lists:flatten(CatsStr) ++ ")\">.*", +                         [unicode]),      lists:flatmap(fun(Str) ->  			  case re:run(Str, MP) of  			      {match,_} -> []; @@ -887,13 +918,13 @@ comment(Comment) when is_list(Comment) ->      Formatted =  	case (catch io_lib:format("~ts",[Comment])) of  	    {'EXIT',_} ->  % it's a list not a string -		io_lib:format("~p",[Comment]); +		io_lib:format("~tp",[Comment]);  	    String ->  		String  	end,      send_html_comment(lists:flatten(Formatted));  comment(Comment) -> -    Formatted = io_lib:format("~p",[Comment]), +    Formatted = io_lib:format("~tp",[Comment]),      send_html_comment(lists:flatten(Formatted)).  %%%----------------------------------------------------------------- diff --git a/lib/common_test/src/ct_config.erl b/lib/common_test/src/ct_config.erl index 99de311570..d48ae830bb 100644 --- a/lib/common_test/src/ct_config.erl +++ b/lib/common_test/src/ct_config.erl @@ -1,7 +1,7 @@  %%--------------------------------------------------------------------  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2010-2016. All Rights Reserved. +%% Copyright Ericsson AB 2010-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -171,8 +171,8 @@ reload_config(KeyOrName) ->  process_default_configs(Opts) ->      lists:flatmap(fun({config,[_|_] = FileOrFiles}) -> -			  case {io_lib:printable_list(FileOrFiles), -				io_lib:printable_list(hd(FileOrFiles))} of +			  case {io_lib:printable_unicode_list(FileOrFiles), +				io_lib:printable_unicode_list(hd(FileOrFiles))} of  			      {false,true} ->  				  FileOrFiles;  			      {true,false} -> diff --git a/lib/common_test/src/ct_conn_log_h.erl b/lib/common_test/src/ct_conn_log_h.erl index 93e64c65fe..cf0a228e1b 100644 --- a/lib/common_test/src/ct_conn_log_h.erl +++ b/lib/common_test/src/ct_conn_log_h.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2012-2016. All Rights Reserved. +%% Copyright Ericsson AB 2012-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -186,7 +186,7 @@ format_head(ConnMod,_,Time,Text) ->      io_lib:format("~n~ts",[Head]).  format_title(raw,#conn_log{client=Client}=Info) -> -    io_lib:format("Client ~w ~s ~ts",[Client,actionstr(Info),serverstr(Info)]); +    io_lib:format("Client ~tw ~s ~ts",[Client,actionstr(Info),serverstr(Info)]);  format_title(_,Info) ->      Title = pad_char_end(?WIDTH,pretty_title(Info),$=),      io_lib:format("~n~ts", [Title]). @@ -197,9 +197,9 @@ format_data(ConnMod,LogType,Data) ->      ConnMod:format_data(LogType,Data).  format_error(raw,Report) -> -    io_lib:format("~n~p~n",[Report]); +    io_lib:format("~n~tp~n",[Report]);  format_error(pretty,Report) -> -    [io_lib:format("~n    ~p: ~p",[K,V]) || {K,V} <- Report]. +    [io_lib:format("~n    ~tp: ~tp",[K,V]) || {K,V} <- Report].  %%%----------------------------------------------------------------- @@ -230,7 +230,7 @@ pretty_head({{{Y,Mo,D},{H,Mi,S}},MicroS},ConnMod,Text0) ->  		   micro2milli(MicroS)]).  pretty_title(#conn_log{client=Client}=Info) -> -    io_lib:format("= Client ~w ~s ~ts ", +    io_lib:format("= Client ~tw ~s ~ts ",  		  [Client,actionstr(Info),serverstr(Info)]).  actionstr(#conn_log{action=send}) -> "----->"; @@ -238,16 +238,18 @@ actionstr(#conn_log{action=cmd}) -> "----->";  actionstr(#conn_log{action=recv}) -> "<-----";  actionstr(#conn_log{action=open}) -> "opened session to";  actionstr(#conn_log{action=close}) -> "closed session to"; +actionstr(#conn_log{action=connect}) -> "connected to"; +actionstr(#conn_log{action=disconnect}) -> "disconnected from";  actionstr(_) -> "<---->".  serverstr(#conn_log{name=undefined,address={undefined,_}}) ->      io_lib:format("server",[]);  serverstr(#conn_log{name=undefined,address=Address}) -> -    io_lib:format("~p",[Address]); +    io_lib:format("~tp",[Address]);  serverstr(#conn_log{name=Alias,address={undefined,_}}) -> -    io_lib:format("~w",[Alias]); +    io_lib:format("~tw",[Alias]);  serverstr(#conn_log{name=Alias,address=Address}) -> -    io_lib:format("~w(~p)",[Alias,Address]). +    io_lib:format("~tw(~tp)",[Alias,Address]).  month(1) -> "Jan";  month(2) -> "Feb"; diff --git a/lib/common_test/src/ct_event.erl b/lib/common_test/src/ct_event.erl index 5fa9f410bf..1a0ee4f3cd 100644 --- a/lib/common_test/src/ct_event.erl +++ b/lib/common_test/src/ct_event.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 2006-2016. All Rights Reserved. +%% Copyright Ericsson AB 2006-2017. All Rights Reserved.  %%   %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -151,7 +151,7 @@ init(RecvPids) ->  %%--------------------------------------------------------------------  handle_event(Event,State=#state{receivers=RecvPids}) ->      print("~n=== ~w ===~n", [?MODULE]), -    print("~w: ~w~n", [Event#event.name,Event#event.data]), +    print("~tw: ~tw~n", [Event#event.name,Event#event.data]),      lists:foreach(fun(Recv) -> report_event(Recv,Event) end, RecvPids),      {ok,State}. diff --git a/lib/common_test/src/ct_framework.erl b/lib/common_test/src/ct_framework.erl index 141c7f5b0a..6066470233 100644 --- a/lib/common_test/src/ct_framework.erl +++ b/lib/common_test/src/ct_framework.erl @@ -312,7 +312,7 @@ add_defaults(Mod,Func, GroupPath) ->  	    end;  	{'EXIT',Reason} ->  	    ErrStr = io_lib:format("~n*** ERROR *** " -				   "~w:suite/0 failed: ~p~n", +				   "~w:suite/0 failed: ~tp~n",  				   [Suite,Reason]),  	    io:format(ErrStr, []),  	    io:format(?def_gl, ErrStr, []), @@ -335,7 +335,7 @@ add_defaults(Mod,Func, GroupPath) ->  		false ->  		    ErrStr = io_lib:format("~n*** ERROR *** "  					   "Invalid return value from " -					   "~w:suite/0: ~p~n", +					   "~w:suite/0: ~tp~n",  					   [Suite,SuiteInfo]),  		    io:format(ErrStr, []),  		    io:format(?def_gl, ErrStr, []), @@ -344,7 +344,7 @@ add_defaults(Mod,Func, GroupPath) ->  	SuiteInfo ->  	    ErrStr = io_lib:format("~n*** ERROR *** "  				   "Invalid return value from " -				   "~w:suite/0: ~p~n", [Suite,SuiteInfo]), +				   "~w:suite/0: ~tp~n", [Suite,SuiteInfo]),  	    io:format(ErrStr, []),  	    io:format(?def_gl, ErrStr, []),  	    {suite0_failed,bad_return_value} @@ -371,7 +371,7 @@ add_defaults1(Mod,Func, GroupPath, SuiteInfo) ->  	{value,{error,BadGr0Val,GrName}} ->  	    Gr0ErrStr = io_lib:format("~n*** ERROR *** "  				      "Invalid return value from " -				      "~w:group(~w): ~p~n", +				      "~w:group(~tw): ~tp~n",  				      [Mod,GrName,BadGr0Val]),  	    io:format(Gr0ErrStr, []),  	    io:format(?def_gl, Gr0ErrStr, []), @@ -393,7 +393,7 @@ add_defaults1(Mod,Func, GroupPath, SuiteInfo) ->  		{error,BadTC0Val} ->  		    TC0ErrStr = io_lib:format("~n*** ERROR *** "  					      "Invalid return value from " -					      "~w:~w/0: ~p~n", +					      "~w:~tw/0: ~tp~n",  					      [Mod,Func,BadTC0Val]),  		    io:format(TC0ErrStr, []),  		    io:format(?def_gl, TC0ErrStr, []), @@ -921,7 +921,7 @@ error_notification(Mod,Func,_Args,{Error,Loc}) ->  	      end,      ErrorStr = case ErrorSpec of  		 {badmatch,Descr} -> -		     Descr1 = lists:flatten(io_lib:format("~P",[Descr,10])), +		     Descr1 = lists:flatten(io_lib:format("~tP",[Descr,10])),  		     if length(Descr1) > 50 ->  			     Descr2 = string:substr(Descr1,1,50),  			     io_lib:format("{badmatch,~ts...}",[Descr2]); @@ -931,15 +931,15 @@ error_notification(Mod,Func,_Args,{Error,Loc}) ->  		 {test_case_failed,Reason} ->  		     case (catch io_lib:format("{test_case_failed,~ts}", [Reason])) of  			 {'EXIT',_} -> -			     io_lib:format("{test_case_failed,~p}", [Reason]); +			     io_lib:format("{test_case_failed,~tp}", [Reason]);  			 Result -> Result  		     end;  		 {'EXIT',_Reason} = EXIT -> -		     io_lib:format("~P", [EXIT,5]); +		     io_lib:format("~tP", [EXIT,5]);  		 {Spec,_Reason} when is_atom(Spec) -> -		     io_lib:format("~w", [Spec]); +		     io_lib:format("~tw", [Spec]);  		 Other -> -		     io_lib:format("~P", [Other,5]) +		     io_lib:format("~tP", [Other,5])  	     end,      ErrorHtml =  	"<font color=\"brown\">" ++ ct_logs:escape_chars(ErrorStr) ++ "</font>", @@ -996,16 +996,16 @@ error_notification(Mod,Func,_Args,{Error,Loc}) ->  	%% if a function specified by all/0 does not exist, we  	%% pick up undef here  	[{LastMod,LastFunc}|_] when ErrorStr == "undef" -> -	    PrintError("~w:~w could not be executed~nReason: ~ts", +	    PrintError("~w:~tw could not be executed~nReason: ~ts",  		     [LastMod,LastFunc,ErrorStr]);  	[{LastMod,LastFunc}|_] -> -	    PrintError("~w:~w failed~nReason: ~ts", [LastMod,LastFunc,ErrorStr]); +	    PrintError("~w:~tw failed~nReason: ~ts", [LastMod,LastFunc,ErrorStr]);  	[{LastMod,LastFunc,LastLine}|_] ->  	    %% print error to console, we are only  	    %% interested in the last executed expression -	    PrintError("~w:~w failed on line ~w~nReason: ~ts", +	    PrintError("~w:~tw failed on line ~w~nReason: ~ts",  		     [LastMod,LastFunc,LastLine,ErrorStr]),  	    case ct_util:read_suite_data({seq,Mod,Func}) of @@ -1178,7 +1178,7 @@ get_all(Mod, ConfTests) ->  	    case ct_util:get_testdata({error_in_suite,Mod}) of  		undefined ->  		    ErrStr = io_lib:format("~n*** ERROR *** " -					   "~w:all/0 failed: ~p~n", +					   "~w:all/0 failed: ~tp~n",  					   [Mod,ExitReason]),  		    io:format(?def_gl, ErrStr, []),  		    %% save the error info so it doesn't get printed twice @@ -1294,8 +1294,8 @@ save_seq(Mod,Seq,SeqTCs,All) ->  check_private(Seq,TCs,All) ->          Bad = lists:filter(fun(TC) -> lists:member(TC,All) end, TCs),      if Bad /= [] -> -	    Reason = io_lib:format("regular test cases not allowed in sequence ~p: " -				   "~p",[Seq,Bad]), +	    Reason = io_lib:format("regular test cases not allowed in sequence ~tp: " +				   "~tp",[Seq,Bad]),  	    throw({error,list_to_atom(lists:flatten(Reason))});         true ->  	    ok @@ -1312,7 +1312,7 @@ check_multiple(Mod,Seq,TCs) ->  		       end,TCs),      if Bad /= [] ->  	    Reason = io_lib:format("test cases found in multiple sequences: " -				   "~p",[Bad]), +				   "~tp",[Bad]),  	    throw({error,list_to_atom(lists:flatten(Reason))});         true ->  	    ok @@ -1340,15 +1340,15 @@ end_per_suite(_Config) ->  %% if the group config functions are missing in the suite,  %% use these instead  init_per_group(GroupName, Config) -> -    ct:comment(io_lib:format("start of ~p", [GroupName])), -    ct_logs:log("TEST INFO", "init_per_group/2 for ~w missing " +    ct:comment(io_lib:format("start of ~tp", [GroupName])), +    ct_logs:log("TEST INFO", "init_per_group/2 for ~tw missing "  		"in suite, using default.",  		[GroupName]),      Config.  end_per_group(GroupName, _) -> -    ct:comment(io_lib:format("end of ~p", [GroupName])), -    ct_logs:log("TEST INFO", "end_per_group/2 for ~w missing " +    ct:comment(io_lib:format("end of ~tp", [GroupName])), +    ct_logs:log("TEST INFO", "end_per_group/2 for ~tw missing "  		"in suite, using default.",  		[GroupName]),      ok. diff --git a/lib/common_test/src/ct_ftp.erl b/lib/common_test/src/ct_ftp.erl index 84e664b387..8effb06e7e 100644 --- a/lib/common_test/src/ct_ftp.erl +++ b/lib/common_test/src/ct_ftp.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 2003-2016. All Rights Reserved. +%% Copyright Ericsson AB 2003-2017. All Rights Reserved.  %%   %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -119,19 +119,19 @@ open(KeyOrName) ->  	_ ->  	    case ct:get_config(KeyOrName) of  		undefined -> -		    log(heading(open,KeyOrName),"Failed: ~p\n", +		    log(heading(open,KeyOrName),"Failed: ~tp\n",  			[{not_available,KeyOrName}]),  		    {error,{not_available,KeyOrName}};  		_ ->  		    case ct:get_config({KeyOrName,username}) of  			undefined -> -			    log(heading(open,KeyOrName),"Failed: ~p\n", +			    log(heading(open,KeyOrName),"Failed: ~tp\n",  				[{not_available,{KeyOrName,username}}]),  			    {error,{not_available,{KeyOrName,username}}};  			Username ->  			    case ct:get_config({KeyOrName,password}) of  				undefined -> -				    log(heading(open,KeyOrName),"Failed: ~p\n", +				    log(heading(open,KeyOrName),"Failed: ~tp\n",  					[{not_available,{KeyOrName,password}}]),  				    {error,{not_available,{KeyOrName,password}}};  				Password -> @@ -145,7 +145,7 @@ open(KeyOrName,Username,Password) ->      log(heading(open,KeyOrName),"",[]),      case ct:get_config({KeyOrName,ftp}) of  	undefined -> -	    log(heading(open,KeyOrName),"Failed: ~p\n", +	    log(heading(open,KeyOrName),"Failed: ~tp\n",  		[{not_available,{KeyOrName,ftp}}]),  	    {error,{not_available,{KeyOrName,ftp}}};  	Addr -> @@ -284,7 +284,7 @@ init(KeyOrName,{IP,Port},{Username,Password}) ->      case ftp_connect(IP,Port,Username,Password) of  	{ok,FtpPid} ->  	    log(heading(init,KeyOrName),  -		"Opened ftp connection:\nIP: ~p\nUsername: ~p\nPassword: ~p\n", +		"Opened ftp connection:\nIP: ~tp\nUsername: ~tp\nPassword: ~p\n",  		[IP,Username,lists:duplicate(length(Password),$*)]),  	    {ok,FtpPid,#state{ftp_pid=FtpPid,target_name=KeyOrName}};  	Error -> @@ -308,28 +308,28 @@ ftp_connect(IP,Port,Username,Password) ->  %% @hidden  handle_msg({send,LocalFile,RemoteFile},State) ->      log(heading(send,State#state.target_name), -	"LocalFile: ~p\nRemoteFile: ~p\n",[LocalFile,RemoteFile]), +	"LocalFile: ~tp\nRemoteFile: ~tp\n",[LocalFile,RemoteFile]),      Result = ftp:send(State#state.ftp_pid,LocalFile,RemoteFile),      {Result,State};  handle_msg({recv,RemoteFile,LocalFile},State) ->      log(heading(recv,State#state.target_name), -	"RemoteFile: ~p\nLocalFile: ~p\n",[RemoteFile,LocalFile]), +	"RemoteFile: ~tp\nLocalFile: ~tp\n",[RemoteFile,LocalFile]),      Result = ftp:recv(State#state.ftp_pid,RemoteFile,LocalFile),      {Result,State};  handle_msg({cd,Dir},State) -> -    log(heading(cd,State#state.target_name),"Dir: ~p\n",[Dir]), +    log(heading(cd,State#state.target_name),"Dir: ~tp\n",[Dir]),      Result = ftp:cd(State#state.ftp_pid,Dir),      {Result,State};  handle_msg({ls,Dir},State) -> -    log(heading(ls,State#state.target_name),"Dir: ~p\n",[Dir]), +    log(heading(ls,State#state.target_name),"Dir: ~tp\n",[Dir]),      Result = ftp:ls(State#state.ftp_pid,Dir),      {Result,State};  handle_msg({type,Type},State) -> -    log(heading(type,State#state.target_name),"Type: ~p\n",[Type]), +    log(heading(type,State#state.target_name),"Type: ~tp\n",[Type]),      Result = ftp:type(State#state.ftp_pid,Type),      {Result,State};  handle_msg({delete,File},State) -> -    log(heading(delete,State#state.target_name),"Delete file: ~p\n",[File]), +    log(heading(delete,State#state.target_name),"Delete file: ~tp\n",[File]),      Result = ftp:delete(State#state.ftp_pid,File),      {Result,State}. @@ -368,7 +368,7 @@ call(Pid,Msg) ->  heading(Function,Name) -> -    io_lib:format("ct_ftp:~w ~p",[Function,Name]). +    io_lib:format("ct_ftp:~tw ~tp",[Function,Name]).  log(Heading,Str,Args) ->      ct_gen_conn:log(Heading,Str,Args). diff --git a/lib/common_test/src/ct_gen_conn.erl b/lib/common_test/src/ct_gen_conn.erl index 8b59d3ab23..badb7c52ae 100644 --- a/lib/common_test/src/ct_gen_conn.erl +++ b/lib/common_test/src/ct_gen_conn.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2003-2016. All Rights Reserved. +%% Copyright Ericsson AB 2003-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -29,13 +29,6 @@  -export([call/2, call/3, return/2, do_within_time/2]).  -export([log/3, start_log/1, cont_log/2, cont_log_no_timestamp/2, end_log/0]). -%%---------------------------------------------------------------------- -%% Exported types -%%---------------------------------------------------------------------- --export_type([server_id/0, -	      target_name/0, -	      key_or_name/0]). -  -ifdef(debug).  -define(dbg,true).  -else. @@ -54,18 +47,6 @@  		  cb_state,  		  ct_util_server}). -%%------------------------------------------------------------------ -%% Type declarations -%%------------------------------------------------------------------ --type server_id() :: atom(). -%% A `ServerId' which exists in a configuration file. --type target_name() :: atom(). -%% A name which is associated to a `server_id()' via a -%% `require' statement or a call to {@link ct:require/2} in the -%% test suite. --type key_or_name() :: server_id() | target_name(). - -  %%%-----------------------------------------------------------------  %%% @spec start(Address,InitData,CallbackMod,Opts) ->  %%%                                     {ok,Handle} | {error,Reason} @@ -266,7 +247,7 @@ do_start(Opts) ->  	    Error;  	{'DOWN',MRef,process,_,Reason} ->  	    log("ct_gen_conn:start", -		"Connection process died: ~p\n", +		"Connection process died: ~tp\n",  		[Reason]),  	    {error,{connection_process_died,Reason}}      end. @@ -346,7 +327,7 @@ loop(Opts) ->  	    case Opts#gen_opts.reconnect of  		true ->  		    log("Connection down!\nOpening new!", -			"Reason: ~p\nAddress: ~p\n", +			"Reason: ~tp\nAddress: ~tp\n",  			[Reason,Opts#gen_opts.address]),  		    case reconnect(Opts) of  			{ok, NewPid, NewState} -> @@ -357,12 +338,12 @@ loop(Opts) ->  			Error ->  			    ct_util:unregister_connection(self()),  			    log("Reconnect failed. Giving up!", -				"Reason: ~p\n", +				"Reason: ~tp\n",  				[Error])  		    end;  		false ->  		    ct_util:unregister_connection(self()), -		    log("Connection closed!","Reason: ~p\n",[Reason]) +		    log("Connection closed!","Reason: ~tp\n",[Reason])  	    end;  	{'EXIT',Pid,Reason} ->  	    case Opts#gen_opts.ct_util_server of @@ -373,8 +354,9 @@ loop(Opts) ->  	    end;  	{stop, From} ->  	    ct_util:unregister_connection(self()), -	    (Opts#gen_opts.callback):terminate(Opts#gen_opts.conn_pid, -					       Opts#gen_opts.cb_state), +            ConnPid = Opts#gen_opts.conn_pid, +            unlink(ConnPid), +	    (Opts#gen_opts.callback):terminate(ConnPid,Opts#gen_opts.cb_state),  	    return(From,ok),  	    ok;  	{{retry,{Error,_Name,CPid,_Msg}}, From} when  @@ -411,8 +393,9 @@ loop(Opts) ->  		    loop(Opts#gen_opts{cb_state=NewState});  		{stop,Reply,NewState} ->  		    ct_util:unregister_connection(self()), -		    (Opts#gen_opts.callback):terminate(Opts#gen_opts.conn_pid, -						       NewState), +                    ConnPid = Opts#gen_opts.conn_pid, +                    unlink(ConnPid), +		    (Opts#gen_opts.callback):terminate(ConnPid,NewState),  		    return(From,Reply)  	    end;  	Msg when Opts#gen_opts.forward==true -> @@ -422,8 +405,9 @@ loop(Opts) ->  		    loop(Opts#gen_opts{cb_state=NewState});  		{stop,NewState} ->  		    ct_util:unregister_connection(self()), -		    (Opts#gen_opts.callback):terminate(Opts#gen_opts.conn_pid, -						       NewState) +                    ConnPid = Opts#gen_opts.conn_pid, +                    unlink(ConnPid), +		    (Opts#gen_opts.callback):terminate(ConnPid,NewState)  	    end      end. diff --git a/lib/common_test/src/ct_groups.erl b/lib/common_test/src/ct_groups.erl index 333151ee1b..2235365a0e 100644 --- a/lib/common_test/src/ct_groups.erl +++ b/lib/common_test/src/ct_groups.erl @@ -210,7 +210,7 @@ find(Mod, _GrNames, _TCs, [BadTerm | _Gs], Known, _Defs, _FindAll) ->  		    "group "++atom_to_list(lists:last(Known))++  			" in "++atom_to_list(Mod)++":groups/0"  	    end,		  -    Term = io_lib:format("~p", [BadTerm]), +    Term = io_lib:format("~tp", [BadTerm]),      E = "Bad term "++lists:flatten(Term)++" in "++Where,      throw({error,list_to_atom(E)}); @@ -447,7 +447,7 @@ make_conf(Mod, Name, Props, TestSpec) ->  	    {false,false} ->  		ct_logs:log("TEST INFO", "init_per_group/2 and "  			    "end_per_group/2 missing for group " -			    "~w in ~w, using default.", +			    "~tw in ~w, using default.",  			    [Name,Mod]),  		{{ct_framework,init_per_group},  		 {ct_framework,end_per_group}, diff --git a/lib/common_test/src/ct_hooks.erl b/lib/common_test/src/ct_hooks.erl index 8cdc6d8c75..f0592a40be 100644 --- a/lib/common_test/src/ct_hooks.erl +++ b/lib/common_test/src/ct_hooks.erl @@ -235,7 +235,7 @@ call([{Hook, call_id, NextFun} | Rest], Config, Meta, Hooks) ->  	call(resort(NewRest,NewHooks,Meta), Config, Meta, NewHooks)      catch Error:Reason ->  	    Trace = erlang:get_stacktrace(), -	    ct_logs:log("Suite Hook","Failed to start a CTH: ~p:~p", +	    ct_logs:log("Suite Hook","Failed to start a CTH: ~tp:~tp",  			[Error,{Reason,Trace}]),  	    call([], {fail,"Failed to start CTH"  		      ", see the CT Log for details"}, Meta, Hooks) @@ -424,11 +424,11 @@ catch_apply(M,F,A) ->          erlang:apply(M,F,A)      catch _:Reason ->              Trace = erlang:get_stacktrace(), -            ct_logs:log("Suite Hook","Call to CTH failed: ~w:~p", +            ct_logs:log("Suite Hook","Call to CTH failed: ~w:~tp",                              [error,{Reason,Trace}]),              throw({error_in_cth_call,                     lists:flatten( -                     io_lib:format("~w:~w/~w CTH call failed", +                     io_lib:format("~w:~tw/~w CTH call failed",                                     [M,F,length(A)]))})      end. diff --git a/lib/common_test/src/ct_logs.erl b/lib/common_test/src/ct_logs.erl index 09ad709da5..ba7660fe6a 100644 --- a/lib/common_test/src/ct_logs.erl +++ b/lib/common_test/src/ct_logs.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2003-2016. All Rights Reserved. +%% Copyright Ericsson AB 2003-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -41,6 +41,7 @@  -export([xhtml/2, locate_priv_file/1, make_relative/1]).  -export([insert_javascript/1]).  -export([uri/1]). +-export([parse_keep_logs/1]).  %% Logging stuff directly from testcase  -export([tc_log/3, tc_log/4, tc_log/5, tc_log/6, @@ -138,7 +139,7 @@ close(Info, StartDir) ->      LogCacheBin =   	case make_last_run_index() of  	    {error, Reason} ->  % log server not responding -		io:format("Warning! ct_logs not responding: ~p~n", [Reason]), +		io:format("Warning! ct_logs not responding: ~tp~n", [Reason]),  		undefined;  	    LCB ->  		LCB @@ -174,7 +175,7 @@ close(Info, StartDir) ->  		ok ->  		    ok;  		Error -> -		    io:format("Warning! Cleanup failed: ~p~n", [Error]) +		    io:format("Warning! Cleanup failed: ~tp~n", [Error])  	    end,  	    _ = make_all_suites_index(stop),  	    make_all_runs_index(stop), @@ -424,7 +425,7 @@ add_external_logs(Logs) ->  %%% @doc Print a link to a given file stored in the priv_dir of the  %%% calling test suite.  add_link(Heading,File,Type) -> -    log(Heading,"<a href=\"~ts\" type=~p>~ts</a>\n", +    log(Heading,"<a href=\"~ts\" type=~tp>~ts</a>\n",  	[uri(filename:join("log_private",File)),Type,File]). @@ -566,7 +567,7 @@ get_header("default") ->  		  [log_timestamp(?now)]);  get_header(Heading) ->      io_lib:format("\n-----------------------------" -		  "-----------------------\n~s ~s\n", +		  "-----------------------\n~ts ~s\n",  		  [Heading,log_timestamp(?now)]).     @@ -703,8 +704,8 @@ logger(Parent, Mode, Verbosity) ->  	    case copy_priv_files(PrivFilesSrc, PrivFilesDestTop) of  		{error,Src1,Dest1,Reason1} ->  		    io:format(?def_gl, "ERROR! "++ -				  "Priv file ~p could not be copied to ~p. "++ -				  "Reason: ~p~n", +				  "Priv file ~tp could not be copied to ~tp. "++ +				  "Reason: ~tp~n",  			      [Src1,Dest1,Reason1]),  		    exit({priv_file_error,Dest1});  		ok -> @@ -712,8 +713,8 @@ logger(Parent, Mode, Verbosity) ->  			{error,Src2,Dest2,Reason2} ->  			    io:format(?def_gl,  				      "ERROR! "++ -				      "Priv file ~p could not be copied to ~p. " -				      ++"Reason: ~p~n", +				      "Priv file ~tp could not be copied to ~tp. " +				      ++"Reason: ~tp~n",  				      [Src2,Dest2,Reason2]),  			    exit({priv_file_error,Dest2});  			ok -> @@ -890,7 +891,7 @@ logger_loop(State) ->  	    logger_loop(State);  	{set_stylesheet,TC,SSFile} ->  	    Fd = State#logger_state.ct_log_fd, -	    io:format(Fd, "~p loading external style sheet: ~ts~n", +	    io:format(Fd, "~tp loading external style sheet: ~ts~n",  		      [TC,SSFile]),  	    logger_loop(State#logger_state{stylesheet = SSFile});  	{clear_stylesheet,_} when State#logger_state.stylesheet == undefined -> @@ -951,7 +952,7 @@ create_io_fun(FromPid, CtLogFd, EscChars) ->  		    [IoList,"\n",IoStr]  	    catch  		_:_Reason -> -		    io:format(CtLogFd, "Logging fails! Str: ~p, Args: ~p~n", +		    io:format(CtLogFd, "Logging fails! Str: ~tp, Args: ~tp~n",  			      [Str,Args]),  		    %% stop the testcase, we need to see the fault  		    exit(FromPid, {log_printout_error,Str,Args}), @@ -1150,7 +1151,7 @@ open_ctlog(MiscIoName) ->  	    Dir = filename:dirname(Cwd),  	    Variables = ct_run:variables_file_name(Dir),  	    io:format(Fd, -		      "Can not read the file \'~ts\' Reason: ~w\n" +		      "Can not read the file \'~ts\' Reason: ~tw\n"  		      "No configuration found for test!!\n",  		      [Variables,Reason])      end, @@ -1212,7 +1213,7 @@ print_style(Fd, IoFormat, StyleSheet) ->      end.  print_style_error(Fd, IoFormat, StyleSheet, Reason) -> -    IO = io_lib:format("\n<!-- Failed to load stylesheet ~ts: ~p -->\n", +    IO = io_lib:format("\n<!-- Failed to load stylesheet ~ts: ~tp -->\n",  		       [StyleSheet,Reason]),      IoFormat(Fd, IO, []),      print_style(Fd, IoFormat, undefined). @@ -1255,11 +1256,11 @@ make_last_run_index(StartTime) ->  	case catch make_last_run_index1(StartTime,IndexName) of  	    {'EXIT', Reason} ->  		io:put_chars("CRASHED while updating " ++ AbsIndexName ++ "!\n"), -		io:format("~p~n", [Reason]), +		io:format("~tp~n", [Reason]),  		{error, Reason};  	    {error, Reason} ->  		io:put_chars("FAILED while updating " ++ AbsIndexName ++ "\n"), -		io:format("~p~n", [Reason]), +		io:format("~tp~n", [Reason]),  		{error, Reason};  	    ok ->  		ok; @@ -1560,7 +1561,7 @@ get_missing_suites(_,_) ->      [].  term_to_text(Term) -> -    lists:flatten(io_lib:format("~p.\n", [Term])). +    lists:flatten(io_lib:format("~tp.\n", [Term])).  %%% Headers and footers. @@ -1828,7 +1829,7 @@ count_cases(Dir) ->  			    Summary  		    end;  		{error, Reason} -> -		    io:format("\nFailed to read ~p: ~p (skipped)\n", +		    io:format("\nFailed to read ~tp: ~tp (skipped)\n",  			      [LogFile,Reason]),  		    error  	    end @@ -1910,10 +1911,10 @@ config_table_header() ->  config_table1([{Key,Value}|Vars]) ->      [xhtml(["<tr><td>", atom_to_list(Key), "</td>\n", -	   "<td><pre>",io_lib:format("~p",[Value]),"</pre></td></tr>\n"], +	   "<td><pre>",io_lib:format("~tp",[Value]),"</pre></td></tr>\n"],  	   ["<tr class=\"", odd_or_even(), "\">\n",  	    "<td>", atom_to_list(Key), "</td>\n", -	    "<td>", io_lib:format("~p",[Value]), "</td>\n</tr>\n"]) | +	    "<td>", io_lib:format("~tp",[Value]), "</td>\n</tr>\n"]) |       config_table1(Vars)];  config_table1([]) ->      [xhtml("","</tbody>\n"),"</table>\n"]. @@ -1946,7 +1947,11 @@ make_all_runs_index(When) ->  	end,	      Dirs = filelib:wildcard(logdir_prefix()++"*.*"), -    DirsSorted = (catch sort_all_runs(Dirs)), +    DirsSorted0 = (catch sort_all_runs(Dirs)), +    DirsSorted = +        if When == start -> DirsSorted0; +           true -> maybe_delete_old_dirs(DirsSorted0) +        end,      LogCacheInfo = get_cache_data(UseCache), @@ -2064,6 +2069,36 @@ sort_ct_runs(Dirs) ->  	      {DateHH1,MM1,SS1} =< {DateHH2,MM2,SS2}        end, Dirs). +parse_keep_logs([Str="all"]) -> +    parse_keep_logs(list_to_atom(Str)); +parse_keep_logs([NStr]) -> +    parse_keep_logs(list_to_integer(NStr)); +parse_keep_logs(all) -> +    all; +parse_keep_logs(N) when is_integer(N), N>0 -> +    N. + +maybe_delete_old_dirs(Sorted) -> +    {Keep,Delete} = +        case application:get_env(common_test, keep_logs) of +            {ok,MaxN} when is_integer(MaxN), length(Sorted)>MaxN -> +                lists:split(MaxN,Sorted); +            _ -> +                {Sorted,[]} +        end, +    delete_old_dirs(Delete), +    Keep. + +delete_old_dirs([]) -> +    ok; +delete_old_dirs(Dirs) -> +    io:put_chars("\n  Removing old test directories:\n"), +    [begin +         io:put_chars("    " ++ Dir ++ "\n"), +         rm_dir(Dir) +     end|| Dir <- Dirs], +    ok. +  dir_diff_all_runs(Dirs, LogCache) ->      case LogCache#log_cache.all_runs of  	[] -> @@ -2439,17 +2474,17 @@ make_all_suites_index(NewTestData = {_TestName,DirName}) ->  					   LogDirData) of  	    {'EXIT',Reason} ->  		io:put_chars("CRASHED while updating " ++ AbsIndexName ++ "!\n"), -		io:format("~p~n", [Reason]), +		io:format("~tp~n", [Reason]),  		{error,Reason};  	    {error,Reason} ->  		io:put_chars("FAILED while updating " ++ AbsIndexName ++ "\n"), -		io:format("~p~n", [Reason]), +		io:format("~tp~n", [Reason]),  		{error,Reason};  	    ok ->  		ok;  	    Err ->  		io:format("Unknown internal error while updating ~ts. " -			  "Please report.\n(Err: ~p, ID: 1)", +			  "Please report.\n(Err: ~tp, ID: 1)",  			  [AbsIndexName,Err]),  		{error, Err}  	end, @@ -2668,11 +2703,11 @@ make_all_suites_index1(When, AbsIndexName, AllTestLogDirs) ->      case catch make_all_suites_index2(IndexName, AllTestLogDirs) of  	{'EXIT', Reason} ->  	    io:put_chars("CRASHED while updating " ++ AbsIndexName ++ "!\n"), -	    io:format("~p~n", [Reason]), +	    io:format("~tp~n", [Reason]),  	    {error, Reason};  	{error, Reason} ->  	    io:put_chars("FAILED while updating " ++ AbsIndexName ++ "\n"), -	    io:format("~p~n", [Reason]), +	    io:format("~tp~n", [Reason]),  	    {error, Reason};  	{ok,TempData} ->  	    case When of @@ -2686,7 +2721,7 @@ make_all_suites_index1(When, AbsIndexName, AllTestLogDirs) ->  	    end;  	Err ->  	    io:format("Unknown internal error while updating ~ts. " -		      "Please report.\n(Err: ~p, ID: 1)", +		      "Please report.\n(Err: ~tp, ID: 1)",  		      [AbsIndexName,Err]),  	    {error, Err}      end. @@ -3165,7 +3200,7 @@ get_ts_html_wrapper(TestName, Logdir, PrintLabel, Cwd, TableCols, Encoding) ->      TestName1 = if is_list(TestName) ->  			lists:flatten(TestName);  		   true -> -			lists:flatten(io_lib:format("~p", [TestName])) +			lists:flatten(io_lib:format("~tp", [TestName]))  		end,      Basic = basic_html(),      LabelStr = diff --git a/lib/common_test/src/ct_make.erl b/lib/common_test/src/ct_make.erl index f22959d457..220cb0473d 100644 --- a/lib/common_test/src/ct_make.erl +++ b/lib/common_test/src/ct_make.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 2009-2016. All Rights Reserved. +%% Copyright Ericsson AB 2009-2017. All Rights Reserved.  %%   %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -96,7 +96,7 @@ read_emakefile(Emakefile,Opts) ->  	    Mods = [filename:rootname(F) ||  F <- filelib:wildcard("*.erl")],  	    [{Mods, Opts}];  	{error,Other} -> -	    io:format("make: Trouble reading 'Emakefile':~n~p~n",[Other]), +	    io:format("make: Trouble reading 'Emakefile':~n~tp~n",[Other]),  	    error      end. @@ -151,7 +151,7 @@ get_opts_from_emakefile(Mods,Emakefile,Opts) ->  	{error,enoent} ->  	    [{Mods, Opts}];  	{error,Other} -> -	    io:format("make: Trouble reading 'Emakefile':~n~p~n",[Other]), +	    io:format("make: Trouble reading 'Emakefile':~n~tp~n",[Other]),  	    error      end. @@ -280,15 +280,47 @@ recompile(File, NoExec, Load, Opts) ->  do_recompile(_File, true, _Load, _Opts) ->      out_of_date; -do_recompile(File, false, noload, Opts) -> +do_recompile(File, false, Load, Opts) ->      io:format("Recompile: ~ts\n",[File]), -    compile:file(File, [report_errors, report_warnings, error_summary |Opts]); -do_recompile(File, false, load, Opts) -> -    io:format("Recompile: ~ts\n",[File]), -    c:c(File, Opts); -do_recompile(File, false, netload, Opts) -> -    io:format("Recompile: ~ts\n",[File]), -    c:nc(File, Opts). +    case compile:file(File, [report_errors, report_warnings |Opts]) of +        Ok when is_tuple(Ok), element(1,Ok)==ok -> +            maybe_load(element(2,Ok), Load, Opts); +        _Error -> +            error +    end. + +maybe_load(_Mod, noload, _Opts) -> +    ok; +maybe_load(Mod, Load, Opts) -> +    %% We have compiled File with options Opts. Find out where the +    %% output file went to, and load it. +    case compile:output_generated(Opts) of +        true -> +            Dir = proplists:get_value(outdir,Opts,"."), +            do_load(Dir, Mod, Load); +        false -> +            io:format("** Warning: No object file created - nothing loaded **~n"), +            ok +    end. + +do_load(Dir, Mod, load) -> +    code:purge(Mod), +    case code:load_abs(filename:join(Dir, Mod),Mod) of +        {module,Mod} -> +            {ok,Mod}; +        Other -> +            Other +    end; +do_load(Dir, Mod, netload) -> +    Obj = atom_to_list(Mod) ++ code:objfile_extension(), +    Fname = filename:join(Dir, Obj), +    case file:read_file(Fname) of +        {ok,Bin} -> +            rpc:eval_everywhere(code,load_binary,[Mod,Fname,Bin]), +            {ok,Mod}; +        Other -> +            Other +    end.  exists(File) ->      case file:read_file_info(File) of diff --git a/lib/common_test/src/ct_master.erl b/lib/common_test/src/ct_master.erl index 4eef27d2a5..6e6d1879c2 100644 --- a/lib/common_test/src/ct_master.erl +++ b/lib/common_test/src/ct_master.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2006-2016. All Rights Reserved. +%% Copyright Ericsson AB 2006-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -434,7 +434,7 @@ init_master1(Parent,NodeOptsList,InitOptions,LogDirs) ->  init_master2(Parent,NodeOptsList,LogDirs) ->      process_flag(trap_exit,true),      Cookie = erlang:get_cookie(), -    log(all,"Cookie","~w",[Cookie]), +    log(all,"Cookie","~tw",[Cookie]),      log(all,"Starting Tests",  	"Tests starting on: ~p",[[N || {N,_} <- NodeOptsList]]),      SpawnAndMon =  @@ -454,7 +454,7 @@ master_loop(#state{node_ctrl_pids=[],  		   results=Finished}) ->      Str =  	lists:map(fun({Node,Result}) -> -			  io_lib:format("~-40.40.*ts~p\n", +			  io_lib:format("~-40.40.*ts~tp\n",  					[$_,atom_to_list(Node),Result])  		  end,lists:reverse(Finished)),      log(all,"TEST RESULTS",Str,[]), @@ -488,7 +488,7 @@ master_loop(State=#state{node_ctrl_pids=NodeCtrlPids,  					Bad  				end,  			    log(all,"Test Info", -				"Test on node ~w failed! Reason: ~p", +				"Test on node ~w failed! Reason: ~tp",  				[Node,Error]),  			    {Locks1,Blocked1} =   				update_queue(exit,Node,Locks,Blocked), @@ -501,7 +501,7 @@ master_loop(State=#state{node_ctrl_pids=NodeCtrlPids,  		undefined ->  		    %% ignore (but report) exit from master_logger etc  		    log(all,"Test Info", -			"Warning! Process ~w has terminated. Reason: ~p", +			"Warning! Process ~w has terminated. Reason: ~tp",  			[Pid,Reason]),  			master_loop(State)  	    end; @@ -584,7 +584,7 @@ update_queue(take,Node,From,Lock={Op,Resource},Locks,Blocked) ->      %% Blocked: [{{Operation,Resource},Node,WaitingPid},...]      case lists:keysearch(Lock,1,Locks) of  	{value,{_Lock,Owner}} ->		% other node has lock -	    log(html,"Lock Info","Node ~w blocked on ~w by ~w. Resource: ~p", +	    log(html,"Lock Info","Node ~w blocked on ~w by ~w. Resource: ~tp",  		[Node,Op,Owner,Resource]),  	    Blocked1 = Blocked ++ [{Lock,Node,From}],  	    {Locks,Blocked1}; @@ -599,7 +599,7 @@ update_queue(release,Node,_From,Lock={Op,Resource},Locks,Blocked) ->      case lists:keysearch(Lock,1,Blocked) of  	{value,E={Lock,SomeNode,WaitingPid}} ->  	    Blocked1 = lists:delete(E,Blocked), -	    log(html,"Lock Info","Node ~w proceeds with ~w. Resource: ~p", +	    log(html,"Lock Info","Node ~w proceeds with ~w. Resource: ~tp",  		[SomeNode,Op,Resource]),  	    reply(ok,WaitingPid),		% waiting process may start  	    {Locks1,Blocked1}; @@ -678,7 +678,7 @@ refresh_logs([D|Dirs],Refreshed) ->  refresh_logs([],Refreshed) ->      Str =  	lists:map(fun({D,Result}) -> -			  io_lib:format("Refreshing logs in ~p... ~p", +			  io_lib:format("Refreshing logs in ~tp... ~tp",  					[D,Result])  		  end,Refreshed),      log(all,"Info",Str,[]). @@ -712,7 +712,7 @@ init_node_ctrl(MasterPid,Cookie,Opts) ->      {ok, _} = start_ct_event(),      ct_event:add_handler([{master,MasterPid}]), -    %% log("Running test with options: ~p~n", [Opts]), +    %% log("Running test with options: ~tp~n", [Opts]),      Result = case (catch ct:run_test(Opts)) of  		 ok -> finished_ok;  		 Other -> Other @@ -828,7 +828,7 @@ start_nodes(InitOptions)->  				  "with callback ~w~n", [NodeName,Callback]);  		    {error, Reason, _NodeName} ->  			io:format("Failed to start node ~w with callback ~w! " -				  "Reason: ~p~n", [NodeName, Callback, Reason]) +				  "Reason: ~tp~n", [NodeName, Callback, Reason])  		end;  	    {true, true}->  		io:format("WARNING: Node ~w is alive but has node_start " @@ -857,10 +857,10 @@ eval_on_nodes(InitOptions)->  evaluate(Node, [{M,F,A}|MFAs])->      case rpc:call(Node, M, F, A) of          {badrpc,Reason}-> -	    io:format("WARNING: Failed to call ~w:~w/~w on node ~w " -		      "due to ~p~n", [M,F,length(A),Node,Reason]); +	    io:format("WARNING: Failed to call ~w:~tw/~w on node ~w " +		      "due to ~tp~n", [M,F,length(A),Node,Reason]);  	Result-> -	    io:format("Called ~w:~w/~w on node ~w, result: ~p~n", +	    io:format("Called ~w:~tw/~w on node ~w, result: ~tp~n",  		      [M,F,length(A),Node,Result])      end,      evaluate(Node, MFAs); diff --git a/lib/common_test/src/ct_master_event.erl b/lib/common_test/src/ct_master_event.erl index d28ef42c20..d535d1274e 100644 --- a/lib/common_test/src/ct_master_event.erl +++ b/lib/common_test/src/ct_master_event.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 2006-2016. All Rights Reserved. +%% Copyright Ericsson AB 2006-2017. All Rights Reserved.  %%   %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -71,7 +71,7 @@ stop() ->  	{error,Reason} ->  	    ct_master_logs:log("Error",  			       "No response from CT Master Event.\n" -			       "Reason = ~p\n" +			       "Reason = ~tp\n"  			       "Terminating now!\n",[Reason]),  	    %% communication with event manager fails, kill it  	    catch exit(whereis(?CT_MEVMGR_REF), kill); @@ -135,7 +135,7 @@ handle_event(#event{name=start_logging,node=Node,data=RunDir},State) ->  handle_event(#event{name=Name,node=Node,data=Data},State) ->      print("~n=== ~w ===~n", [?MODULE]), -    print("~w on ~w: ~p~n", [Name,Node,Data]), +    print("~tw on ~w: ~tp~n", [Name,Node,Data]),      {ok,State}.  %%-------------------------------------------------------------------- diff --git a/lib/common_test/src/ct_master_logs.erl b/lib/common_test/src/ct_master_logs.erl index 52003f752d..d8ecd641ed 100644 --- a/lib/common_test/src/ct_master_logs.erl +++ b/lib/common_test/src/ct_master_logs.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2006-2016. All Rights Reserved. +%% Copyright Ericsson AB 2006-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -110,16 +110,16 @@ init(Parent,LogDir,Nodes) ->  	    case copy_priv_files(PrivFilesSrc, PrivFilesDestTop) of  		{error,Src1,Dest1,Reason1} ->  		    io:format(user, "ERROR! "++ -			      "Priv file ~p could not be copied to ~p. "++ -			      "Reason: ~p~n", +			      "Priv file ~tp could not be copied to ~tp. "++ +			      "Reason: ~tp~n",  			      [Src1,Dest1,Reason1]),  		    exit({priv_file_error,Dest1});  		ok ->  		    case copy_priv_files(PrivFilesSrc, PrivFilesDestRun) of  			{error,Src2,Dest2,Reason2} ->  			    io:format(user, "ERROR! "++ -				      "Priv file ~p could not be copied to ~p. "++ -				      "Reason: ~p~n", +				      "Priv file ~tp could not be copied to ~tp. "++ +				      "Reason: ~tp~n",  				      [Src2,Dest2,Reason2]),  			    exit({priv_file_error,Dest2});  			ok -> @@ -170,7 +170,7 @@ loop(State) ->  			case catch io:format(Fd,Str++"\n",Args) of  			    {'EXIT',Reason} ->  				io:format(Fd,  -					  "Logging fails! Str: ~p, Args: ~p~n", +					  "Logging fails! Str: ~tp, Args: ~tp~n",  					  [Str,Args]),  				exit({logging_failed,Reason}),  				ok; diff --git a/lib/common_test/src/ct_master_status.erl b/lib/common_test/src/ct_master_status.erl index 7d3e54e645..b27fdd341e 100644 --- a/lib/common_test/src/ct_master_status.erl +++ b/lib/common_test/src/ct_master_status.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 2006-2016. All Rights Reserved. +%% Copyright Ericsson AB 2006-2017. All Rights Reserved.  %%   %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -71,7 +71,7 @@ init(_) ->  %%  handle_event(#event{name=Name,node=Node,data=Data},State) ->      print("~n=== ~w ===~n", [?MODULE]), -    print("~w on ~w: ~p~n", [Name,Node,Data]), +    print("~tw on ~w: ~tp~n", [Name,Node,Data]),      {ok,State}.  %%-------------------------------------------------------------------- diff --git a/lib/common_test/src/ct_netconfc.erl b/lib/common_test/src/ct_netconfc.erl index ff45407fe0..2c4b97df20 100644 --- a/lib/common_test/src/ct_netconfc.erl +++ b/lib/common_test/src/ct_netconfc.erl @@ -1,7 +1,7 @@  %%----------------------------------------------------------------------  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2012-2016. All Rights Reserved. +%% Copyright Ericsson AB 2012-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -23,24 +23,15 @@  %% Description:  %%    This file contains the Netconf client interface  %% -%% @author Support +%% Netconf servers can be configured by adding the following statement +%% to a configuration file:  %% -%% @doc Netconf client module. +%% {server_id(),options()}.  %% -%% <p>The Netconf client is compliant with RFC4741 and RFC4742.</p> +%% The server_id() or an associated ct:target_name() shall then be +%% used in calls to open/2 connect/2.  %% -%% <p> For each server to test against, the following entry can be -%% added to a configuration file:</p> -%% -%% <p>`{server_id(),options()}.'</p> -%% -%% <p> The `server_id()' or an associated `target_name()' (see -%% {@link ct}) shall then be used in calls to {@link open/2}.</p> -%% -%% <p>If no configuration exists for a server, a session can still be -%% opened by calling {@link open/2} with all necessary options given -%% in the call. The first argument to {@link open/2} can then be any -%% atom.</p> +%% If no configuration exists for a server, use open/1 and connect/1.  %%  %% == Logging ==  %% @@ -49,102 +40,15 @@  %% `ct_conn_log_h'. To use this error handler, add the `cth_conn_log'  %% hook in your test suite, e.g.  %% -%% ```  %% suite() -> -%%    [{ct_hooks, [{cth_conn_log, [{conn_mod(),hook_options()}]}]}]. -%%''' -%% -%% The `conn_mod()' is the name of the common_test module implementing -%% the connection protocol, e.g. `ct_netconfc'. -%% -%% The hook option `log_type' specifies the type of logging: -%% -%% <dl> -%%   <dt>`raw'</dt> -%%   <dd>The sent and received netconf data is logged to a separate -%%   text file as is without any formatting. A link to the file is -%%   added to the test case HTML log.</dd> -%% -%%   <dt>`pretty'</dt> -%%   <dd>The sent and received netconf data is logged to a separate -%%   text file with XML data nicely indented. A link to the file is -%%   added to the test case HTML log.</dd> -%% -%%   <dt>`html (default)'</dt> -%%   <dd>The sent and received netconf traffic is pretty printed -%%   directly in the test case HTML log.</dd> -%% -%%   <dt>`silent'</dt> -%%   <dd>Netconf traffic is not logged.</dd> -%% </dl> -%% -%% By default, all netconf traffic is logged in one single log -%% file. However, it is possible to have different connections logged -%% in separate files. To do this, use the hook option `hosts' and -%% list the names of the servers/connections that will be used in the -%% suite. Note that the connections must be named for this to work, -%% i.e. they must be opened with {@link open/2}. -%% -%% The `hosts' option has no effect if `log_type' is set to `html' or -%% `silent'. -%% -%% The hook options can also be specified in a configuration file with -%% the configuration variable `ct_conn_log': -%% -%% ``` -%% {ct_conn_log,[{conn_mod(),hook_options()}]}. -%% ''' +%%     [{ct_hooks, [{cth_conn_log, [{ct:conn_log_mod(),ct:conn_log_options()}]}]}].  %%  %% For example:  %% -%% ``` -%% {ct_conn_log,[{ct_netconfc,[{log_type,pretty}, -%%                             {hosts,[key_or_name()]}]}]} -%% ''' -%% -%% <b>Note</b> that hook options specified in a configuration file -%% will overwrite the hardcoded hook options in the test suite. -%% -%% === Logging example 1 === -%% -%% The following `ct_hooks' statement will cause pretty printing of -%% netconf traffic to separate logs for the connections named -%% `nc_server1' and `nc_server2'. Any other connections will be logged -%% to default netconf log. -%% -%% ```  %% suite() -> -%%    [{ct_hooks, [{cth_conn_log, [{ct_netconfc,[{log_type,pretty}}, -%%                                               {hosts,[nc_server1,nc_server2]}]} -%%                                ]}]}]. -%%''' -%% -%% Connections must be opened like this: -%% -%% ``` -%% open(nc_server1,[...]), -%% open(nc_server2,[...]). -%% ''' -%% -%% === Logging example 2 === -%% -%% The following configuration file will cause raw logging of all -%% netconf traffic into one single text file. -%% -%% ``` -%% {ct_conn_log,[{ct_netconfc,[{log_type,raw}]}]}. -%% ''' -%% -%% The `ct_hooks' statement must look like this: -%% -%% ``` -%% suite() -> -%%    [{ct_hooks, [{cth_conn_log, []}]}]. -%% ''' -%% -%% The same `ct_hooks' statement without the configuration file would -%% cause HTML logging of all netconf connections into the test case -%% HTML log. +%%     [{ct_hooks, +%%         [{cth_conn_log,[{ct_netconfc,[{log_type,pretty}, +%%                                       {hosts,[my_configured_server]}]}]}  %%  %% == Notifications ==  %% @@ -152,11 +56,9 @@  %% Notifications, which defines a mechanism for an asynchronous  %% message notification delivery service for the netconf protocol.  %% -%% Specific functions to support this are {@link -%% create_subscription/6} and {@link get_event_streams/3}. (The -%% functions also exist with other arities.) +%% Specific functions to support this are create_subscription/6 +%% get_event_streams/3. (The functions also exist with other arities.)  %% -%% @end  %%----------------------------------------------------------------------  -module(ct_netconfc). @@ -167,7 +69,13 @@  %%----------------------------------------------------------------------  %% External exports  %%---------------------------------------------------------------------- --export([open/1, +-export([connect/1, +         connect/2, +         disconnect/1, +         session/1, +         session/2, +         session/3, +         open/1,  	 open/2,  	 only_open/1,  	 only_open/2, @@ -205,6 +113,7 @@  	 create_subscription/4,  	 create_subscription/5,  	 create_subscription/6, +	 get_event_streams/1,  	 get_event_streams/2,  	 get_event_streams/3,  	 get_capabilities/1, @@ -215,7 +124,9 @@  %%----------------------------------------------------------------------  %% Exported types  %%---------------------------------------------------------------------- --export_type([notification/0]). +-export_type([client/0, +              handle/0, +              notification/0]).  %%----------------------------------------------------------------------  %% Internal exports @@ -273,13 +184,15 @@  		  host,  		  port = ?DEFAULT_PORT,  		  timeout = ?DEFAULT_TIMEOUT, -		  name}). +		  name, +                  type}).  %% Connection reference  -record(connection, {reference, % {CM,Ch}  		     host,  		     port, -		     name}). +		     name, +                     type}).  %% Pending replies from server  -record(pending, {tref,    % timer ref (returned from timer:xxx) @@ -291,17 +204,17 @@  %%----------------------------------------------------------------------  %% Type declarations  %%---------------------------------------------------------------------- --type client() :: handle() | ct_gen_conn:server_id() | ct_gen_conn:target_name(). --type handle() :: term(). -%% An opaque reference for a connection (netconf session). See {@link -%% ct} for more information. +-type client() :: handle() | server_id() | ct:target_name(). +-opaque handle() :: pid().  -type options() :: [option()]. -%% Options used for setting up ssh connection to a netconf server. -  -type option() :: {ssh,host()} | {port,inet:port_number()} | {user,string()} |  		  {password,string()} | {user_dir,string()} |  		  {timeout,timeout()}. + +-type session_options() :: [session_option()]. +-type session_option() :: {timeout,timeout()}. +  -type host() :: inet:hostname() | inet:ip_address().  -type notification() :: {notification, xml_attributes(), notification_content()}. @@ -317,14 +230,13 @@  %% See XML Schema for Event Notifications found in RFC5277 for further  %% detail about the data format for the string values. -%-type error_handler() :: module().  -type error_reason() :: term(). +-type server_id() :: atom(). +  -type simple_xml() :: {xml_tag(), xml_attributes(), xml_content()} |  		      {xml_tag(), xml_content()} |  		      xml_tag(). -%% <p>This type is further described in the documentation for the -%% <tt>Xmerl</tt> application.</p>  -type xml_tag() :: atom().  -type xml_attributes() :: [{xml_attribute_tag(),xml_attribute_value()}].  -type xml_attribute_tag() :: atom(). @@ -336,69 +248,130 @@  -type xs_datetime() :: string().  %% This date and time identifyer has the same format as the XML type  %% dateTime and compliant to RFC3339. The format is -%% ```[-]CCYY-MM-DDThh:mm:ss[.s][Z|(+|-)hh:mm]''' +%% "[-]CCYY-MM-DDThh:mm:ss[.s][Z|(+|-)hh:mm]"  %%----------------------------------------------------------------------  %% External interface functions  %%----------------------------------------------------------------------  %%---------------------------------------------------------------------- --spec open(Options) -> Result when +%% Open an SSH connection to a Netconf server +%% If the server options are specified in a configuration file, use +%% open/2. +-spec connect(Options) -> Result when        Options :: options(),        Result :: {ok,handle()} | {error,error_reason()}. -%% @doc Open a netconf session and exchange `hello' messages. -%% -%% If the server options are specified in a configuration file, or if -%% a named client is needed for logging purposes (see {@section -%% Logging}) use {@link open/2} instead. -%% -%% The opaque `handler()' reference which is returned from this -%% function is required as client identifier when calling any other -%% function in this module. -%% -%% The `timeout' option (milli seconds) is used when setting up -%% the ssh connection and when waiting for the hello message from the -%% server. It is not used for any other purposes during the lifetime -%% of the connection. -%% -%% @end +connect(Options) -> +    do_connect(Options, #options{type=connection},[]). + +-spec connect(KeyOrName,ExtraOptions) -> Result when +      KeyOrName :: ct:key_or_name(), +      ExtraOptions :: options(), +      Result :: {ok,handle()} | {error,error_reason()}. +connect(KeyOrName, ExtraOptions) -> +    SortedExtra = lists:keysort(1,ExtraOptions), +    SortedConfig = lists:keysort(1,ct:get_config(KeyOrName,[])), +    AllOpts = lists:ukeymerge(1,SortedConfig,SortedExtra), +    do_connect(AllOpts,#options{name=KeyOrName,type=connection},[{name,KeyOrName}]). + +do_connect(OptList,InitOptRec,NameOpt) -> +    case check_options(OptList,InitOptRec) of +	{Host,Port,Options} -> +	    ct_gen_conn:start({Host,Port},Options,?MODULE, +                              NameOpt ++ [{reconnect,false}, +                                          {use_existing_connection,false}, +                                          {forward_messages,false}]); +	Error -> +	    Error +    end. +  %%---------------------------------------------------------------------- -open(Options) -> -    open(Options,#options{},[],true). +%% Close the given SSH connection. +-spec disconnect(Conn) -> ok | {error,error_reason()} when +      Conn :: handle(). +disconnect(Conn) -> +    case call(Conn,get_ssh_connection) of +        {ok,_} -> +            ct_gen_conn:stop(Conn); +        Error -> +            Error +    end.  %%---------------------------------------------------------------------- +%% Open a netconf session as a channel on the given SSH connection, +%% and exchange `hello' messages. +-spec session(Conn) -> Result when +      Conn :: handle(), +      Result :: {ok,handle()} | {error,error_reason()}. +session(Conn) -> +    do_session(Conn,[],#options{type=channel},[]). + +-spec session(Conn,Options) -> Result when +      Conn :: handle(), +      Options :: session_options(), +      Result :: {ok,handle()} | {error,error_reason()}; +             (KeyOrName,Conn) -> Result when +      KeyOrName :: ct:key_or_name(), +      Conn :: handle(), +      Result :: {ok,handle()} | {error,error_reason()}. +session(Conn,Options) when is_list(Options) -> +    do_session(Conn,Options,#options{type=channel},[]); +session(KeyOrName,Conn) -> +    do_session(Conn,[],#options{name=KeyOrName,type=channel},[{name,KeyOrName}]). + +-spec session(KeyOrName,Conn,Options) -> Result when +      Conn :: handle(), +      Options :: session_options(), +      KeyOrName :: ct:key_or_name(), +      Result :: {ok,handle()} | {error,error_reason()}. +session(KeyOrName,Conn,ExtraOptions) -> +    SortedExtra = lists:keysort(1,ExtraOptions), +    SortedConfig = lists:keysort(1,ct:get_config(KeyOrName,[])), +    AllOpts = lists:ukeymerge(1,SortedConfig,SortedExtra), +    do_session(Conn,AllOpts,#options{name=KeyOrName,type=channel}, +               [{name,KeyOrName}]). + +do_session(Conn,OptList,InitOptRec,NameOpt) -> +    case call(Conn,get_ssh_connection) of +        {ok,SshConn} -> +            case check_session_options(OptList,InitOptRec) of +                {ok,Options} -> +                    case ct_gen_conn:start(SshConn,Options,?MODULE, +                                           NameOpt ++ +                                               [{reconnect,false}, +                                                {use_existing_connection,false}, +                                                {forward_messages,true}]) of +                        {ok,Client} -> +                            case hello(Client,Options#options.timeout) of +                                ok -> +                                    {ok,Client}; +                                Error -> +                                    Error +                            end; +                        Error -> +                            Error +                    end; +                Error -> +                    Error +            end; +	Error -> +	    Error +    end. + +%%---------------------------------------------------------------------- +%% Open a netconf session and exchange 'hello' messages. +%% If the server options are specified in a configuration file, use +%% open/2. +-spec open(Options) -> Result when +      Options :: options(), +      Result :: {ok,handle()} | {error,error_reason()}. +open(Options) -> +    open(Options,#options{type=connection_and_channel},[],true). +  -spec open(KeyOrName, ExtraOptions) -> Result when -      KeyOrName :: ct_gen_conn:key_or_name(), +      KeyOrName :: ct:key_or_name(),        ExtraOptions :: options(),        Result :: {ok,handle()} | {error,error_reason()}. -%% @doc Open a named netconf session and exchange `hello' messages. -%% -%% If `KeyOrName' is a configured `server_id()' or a -%% `target_name()' associated with such an ID, then the options -%% for this server will be fetched from the configuration file. -% -%% The `ExtraOptions' argument will be added to the options found in -%% the configuration file. If the same options are given, the values -%% from the configuration file will overwrite `ExtraOptions'. -%% -%% If the server is not specified in a configuration file, use {@link -%% open/1} instead. -%% -%% The opaque `handle()' reference which is returned from this -%% function can be used as client identifier when calling any other -%% function in this module. However, if `KeyOrName' is a -%% `target_name()', i.e. if the server is named via a call to -%% `ct:require/2' or a `require' statement in the test -%% suite, then this name may be used instead of the `handle()'. -%% -%% The `timeout' option (milli seconds) is used when setting up -%% the ssh connection and when waiting for the hello message from the -%% server. It is not used for any other purposes during the lifetime -%% of the connection. -%% -%% @see ct:require/2 -%% @end -%%----------------------------------------------------------------------  open(KeyOrName, ExtraOpts) ->      open(KeyOrName, ExtraOpts, true). @@ -406,10 +379,11 @@ open(KeyOrName, ExtraOpts, Hello) ->      SortedExtra = lists:keysort(1,ExtraOpts),      SortedConfig = lists:keysort(1,ct:get_config(KeyOrName,[])),      AllOpts = lists:ukeymerge(1,SortedConfig,SortedExtra), -    open(AllOpts,#options{name=KeyOrName},[{name,KeyOrName}],Hello). +    open(AllOpts,#options{name=KeyOrName,type=connection_and_channel}, +         [{name,KeyOrName}],Hello).  open(OptList,InitOptRec,NameOpt,Hello) -> -    case check_options(OptList,undefined,undefined,InitOptRec) of +    case check_options(OptList,InitOptRec) of  	{Host,Port,Options} ->  	    case ct_gen_conn:start({Host,Port},Options,?MODULE,  				   NameOpt ++ [{reconnect,false}, @@ -431,296 +405,206 @@ open(OptList,InitOptRec,NameOpt,Hello) ->  %%---------------------------------------------------------------------- +%% As open/1,2, except no 'hello' message is sent.  -spec only_open(Options) -> Result when        Options :: options(),        Result :: {ok,handle()} | {error,error_reason()}. -%% @doc Open a netconf session, but don't send `hello'. -%% -%% As {@link open/1} but does not send a `hello' message. -%% -%% @end -%%----------------------------------------------------------------------  only_open(Options) -> -    open(Options,#options{},[],false). +    open(Options,#options{type=connection_and_channel},[],false). -%%----------------------------------------------------------------------  -spec only_open(KeyOrName,ExtraOptions) -> Result when -      KeyOrName :: ct_gen_conn:key_or_name(), +      KeyOrName :: ct:key_or_name(),        ExtraOptions :: options(),        Result :: {ok,handle()} | {error,error_reason()}. -%% @doc Open a name netconf session, but don't send `hello'. -%% -%% As {@link open/2} but does not send a `hello' message. -%% -%% @end -%%----------------------------------------------------------------------  only_open(KeyOrName, ExtraOpts) ->      open(KeyOrName, ExtraOpts, false).  %%---------------------------------------------------------------------- -%% @spec hello(Client) -> Result -%% @equiv hello(Client, [], infinity) +%% Send a 'hello' message. +-spec hello(Client) -> Result when +      Client :: handle(), +      Result :: ok | {error,error_reason()}.  hello(Client) ->      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. -%% -%% Adds optional capabilities and sends a `hello' message to the -%% server and waits for the return. -%% @end -%%----------------------------------------------------------------------  hello(Client,Options,Timeout) ->      call(Client, {hello, Options, Timeout}).  %%---------------------------------------------------------------------- -%% @spec get_session_id(Client) -> Result -%% @equiv get_session_id(Client, infinity) +%% Get the session id for the session specified by Client. +-spec get_session_id(Client) -> Result when +      Client :: client(), +      Result :: pos_integer() | {error,error_reason()}.  get_session_id(Client) ->      get_session_id(Client, ?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec get_session_id(Client, Timeout) -> Result when        Client :: client(),        Timeout :: timeout(),        Result :: pos_integer() | {error,error_reason()}. -%% @doc Returns the session id associated with the given client. -%% -%% @end -%%----------------------------------------------------------------------  get_session_id(Client, Timeout) ->      call(Client, get_session_id, Timeout).  %%---------------------------------------------------------------------- -%% @spec get_capabilities(Client) -> Result -%% @equiv get_capabilities(Client, infinity) +%% Get the server side capabilities. +-spec get_capabilities(Client) -> Result when +      Client :: client(), +      Result :: [string()] | {error,error_reason()}.  get_capabilities(Client) ->      get_capabilities(Client, ?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec get_capabilities(Client, Timeout) -> Result when        Client :: client(),        Timeout :: timeout(),        Result :: [string()] | {error,error_reason()}. -%% @doc Returns the server side capabilities -%% -%% The following capability identifiers, defined in RFC 4741, can be returned: -%% -%% <ul> -%%   <li>`"urn:ietf:params:netconf:base:1.0"'</li> -%%   <li>`"urn:ietf:params:netconf:capability:writable-running:1.0"'</li> -%%   <li>`"urn:ietf:params:netconf:capability:candidate:1.0"'</li> -%%   <li>`"urn:ietf:params:netconf:capability:confirmed-commit:1.0"'</li> -%%   <li>`"urn:ietf:params:netconf:capability:rollback-on-error:1.0"'</li> -%%   <li>`"urn:ietf:params:netconf:capability:startup:1.0"'</li> -%%   <li>`"urn:ietf:params:netconf:capability:url:1.0"'</li> -%%   <li>`"urn:ietf:params:netconf:capability:xpath:1.0"'</li> -%% </ul> -%% -%% Note, additional identifiers may exist, e.g. server side namespace. -%% -%% @end -%%----------------------------------------------------------------------  get_capabilities(Client, Timeout) ->      call(Client, get_capabilities, Timeout).  %%---------------------------------------------------------------------- -%% @spec send(Client, SimpleXml) -> Result -%% @equiv send(Client, SimpleXml, infinity) +%% Send an XML document to the server. +-spec send(Client, SimpleXml) -> Result when +      Client :: client(), +      SimpleXml :: simple_xml(), +      Result :: simple_xml() | {error,error_reason()}.  send(Client, SimpleXml) ->      send(Client, SimpleXml, ?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec send(Client, SimpleXml, Timeout) -> Result when        Client :: client(),        SimpleXml :: simple_xml(),        Timeout :: timeout(),        Result :: simple_xml() | {error,error_reason()}. -%% @doc Send an XML document to the server. -%% -%% The given XML document is sent as is to the server. This function -%% can be used for sending XML documents that can not be expressed by -%% other interface functions in this module.  send(Client, SimpleXml, Timeout) ->      call(Client,{send, Timeout, SimpleXml}).  %%---------------------------------------------------------------------- -%% @spec send_rpc(Client, SimpleXml) -> Result -%% @equiv send_rpc(Client, SimpleXml, infinity) +%% Wrap the given XML document in a valid netconf 'rpc' request and +%% send to the server. +-spec send_rpc(Client, SimpleXml) -> Result when +      Client :: client(), +      SimpleXml :: simple_xml(), +      Result :: [simple_xml()] | {error,error_reason()}.  send_rpc(Client, SimpleXml) ->      send_rpc(Client, SimpleXml, ?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec send_rpc(Client, SimpleXml, Timeout) -> Result when        Client :: client(),        SimpleXml :: simple_xml(),        Timeout :: timeout(),        Result :: [simple_xml()] | {error,error_reason()}. -%% @doc Send a Netconf <code>rpc</code> request to the server. -%% -%% The given XML document is wrapped in a valid Netconf -%% <code>rpc</code> request and sent to the server. The -%% <code>message-id</code> and namespace attributes are added to the -%% <code>rpc</code> element. -%% -%% This function can be used for sending <code>rpc</code> requests -%% that can not be expressed by other interface functions in this -%% module.  send_rpc(Client, SimpleXml, Timeout) ->      call(Client,{send_rpc, SimpleXml, Timeout}).  %%---------------------------------------------------------------------- -%% @spec lock(Client, Target) -> Result -%% @equiv lock(Client, Target, infinity) +%% Send a 'lock' request. +-spec lock(Client, Target) -> Result when +      Client :: client(), +      Target :: netconf_db(), +      Result :: ok | {error,error_reason()}.  lock(Client, Target) ->      lock(Client, Target,?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec lock(Client, Target, Timeout) -> Result when        Client :: client(),        Target :: netconf_db(),        Timeout :: timeout(),        Result :: ok | {error,error_reason()}. -%% @doc Unlock configuration target. -%% -%% Which target parameters that can be used depends on if -%% `:candidate' and/or `:startup' are supported by the -%% server. If successfull, the configuration system of the device is -%% not available to other clients (Netconf, CORBA, SNMP etc). Locks -%% are intended to be short-lived. -%% -%% The operations {@link kill_session/2} or {@link kill_session/3} can -%% be used to force the release of a lock owned by another Netconf -%% session. How this is achieved by the server side is implementation -%% specific. -%% -%% @end -%%----------------------------------------------------------------------  lock(Client, Target, Timeout) ->      call(Client,{send_rpc_op,lock,[Target],Timeout}).  %%---------------------------------------------------------------------- -%% @spec unlock(Client, Target) -> Result -%% @equiv unlock(Client, Target, infinity) +%% Send a 'unlock' request. +-spec unlock(Client, Target) -> Result when +      Client :: client(), +      Target :: netconf_db(), +      Result :: ok | {error,error_reason()}.  unlock(Client, Target) ->      unlock(Client, Target,?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec unlock(Client, Target, Timeout) -> Result when        Client :: client(),        Target :: netconf_db(),        Timeout :: timeout(),        Result :: ok | {error,error_reason()}. -%% @doc Unlock configuration target. -%% -%% If the client earlier has aquired a lock, via {@link lock/2} or -%% {@link lock/3}, this operation release the associated lock.  To be -%% able to access another target than `running', the server must -%% support `:candidate' and/or `:startup'. -%% -%% @end -%%----------------------------------------------------------------------  unlock(Client, Target, Timeout) ->      call(Client, {send_rpc_op, unlock, [Target], Timeout}).  %%---------------------------------------------------------------------- -%% @spec get(Client, Filter) -> Result -%% @equiv get(Client, Filter, infinity) +%% Send a 'get' request. +-spec get(Client, Filter) -> Result when +      Client :: client(), +      Filter :: simple_xml() | xpath(), +      Result :: {ok,[simple_xml()]} | {error,error_reason()}.  get(Client, Filter) ->      get(Client, Filter, ?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec get(Client, Filter, Timeout) -> Result when        Client :: client(),        Filter :: simple_xml() | xpath(),        Timeout :: timeout(),        Result :: {ok,[simple_xml()]} | {error,error_reason()}. -%% @doc Get data. -%% -%% This operation returns both configuration and state data from the -%% server. -%% -%% Filter type `xpath' can only be used if the server supports -%% `:xpath'. -%% -%% @end -%%----------------------------------------------------------------------  get(Client, Filter, Timeout) ->      call(Client,{send_rpc_op, get, [Filter], Timeout}).  %%---------------------------------------------------------------------- -%% @spec get_config(Client, Source, Filter) -> Result -%% @equiv get_config(Client, Source, Filter, infinity) +%% Send a 'get-config' request. +-spec get_config(Client, Source, Filter) -> Result when +      Client :: client(), +      Source :: netconf_db(), +      Filter :: simple_xml() | xpath(), +      Result :: {ok,[simple_xml()]} | {error,error_reason()}.  get_config(Client, Source, Filter) ->      get_config(Client, Source, Filter, ?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec get_config(Client, Source, Filter, Timeout) -> Result when        Client :: client(),        Source :: netconf_db(),        Filter :: simple_xml() | xpath(),        Timeout :: timeout(),        Result :: {ok,[simple_xml()]} | {error,error_reason()}. -%% @doc Get configuration data. -%% -%% To be able to access another source than `running', the server -%% must advertise `:candidate' and/or `:startup'. -%% -%% Filter type `xpath' can only be used if the server supports -%% `:xpath'. -%% -%% -%% @end -%%----------------------------------------------------------------------  get_config(Client, Source, Filter, Timeout) ->      call(Client, {send_rpc_op, get_config, [Source, Filter], Timeout}).  %%---------------------------------------------------------------------- -%% @spec edit_config(Client, Target, Config) -> Result -%% @equiv edit_config(Client, Target, Config, [], infinity) +%% Send a 'edit-config' request. +-spec edit_config(Client, Target, Config) -> Result when +      Client :: client(), +      Target :: netconf_db(), +      Config :: simple_xml(), +      Result :: ok | {error,error_reason()}.  edit_config(Client, Target, Config) ->      edit_config(Client, Target, Config, ?DEFAULT_TIMEOUT). -%%---------------------------------------------------------------------- --spec edit_config(Client, Target, Config, OptParamsOrTimeout) -> Result when +-spec edit_config(Client, Target, Config, OptParams) -> Result when +      Client :: client(), +      Target :: netconf_db(), +      Config :: simple_xml(), +      OptParams :: [simple_xml()], +      Result :: ok | {error,error_reason()}; +                 (Client, Target, Config, Timeout) -> Result when        Client :: client(),        Target :: netconf_db(),        Config :: simple_xml(), -      OptParamsOrTimeout :: [simple_xml()] | timeout(), +      Timeout :: 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(), @@ -728,99 +612,79 @@ edit_config(Client, Target, Config, OptParams) when is_list(OptParams) ->        OptParams :: [simple_xml()],        Timeout :: timeout(),        Result :: ok | {error,error_reason()}. -%% @doc Edit configuration data. -%% -%% Per default only the running target is available, unless the server -%% 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, OptParams, Timeout) ->      call(Client, {send_rpc_op, edit_config, [Target,Config,OptParams], Timeout}).  %%---------------------------------------------------------------------- -%% @spec delete_config(Client, Target) -> Result -%% @equiv delete_config(Client, Target, infinity) +%% Send a 'delete-config' request. +-spec delete_config(Client, Target) -> Result when +      Client :: client(), +      Target :: startup | candidate, +      Result :: ok | {error,error_reason()}.  delete_config(Client, Target) ->      delete_config(Client, Target, ?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec delete_config(Client, Target, Timeout) -> Result when        Client :: client(),        Target :: startup | candidate,        Timeout :: timeout(),        Result :: ok | {error,error_reason()}. -%% @doc Delete configuration data. -%% -%% The running configuration cannot be deleted and `:candidate' -%% or `:startup' must be advertised by the server. -%% -%% @end -%%----------------------------------------------------------------------  delete_config(Client, Target, Timeout) when Target == startup;  					    Target == candidate ->      call(Client,{send_rpc_op, delete_config, [Target], Timeout}).  %%---------------------------------------------------------------------- -%% @spec copy_config(Client, Source, Target) -> Result -%% @equiv copy_config(Client, Source, Target, infinity) +%% Send a 'copy-config' request. +-spec copy_config(Client, Target, Source) -> Result when +      Client :: client(), +      Target :: netconf_db(), +      Source :: netconf_db(), +      Result :: ok | {error,error_reason()}.  copy_config(Client, Source, Target) ->      copy_config(Client, Source, Target, ?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec copy_config(Client, Target, Source, Timeout) -> Result when        Client :: client(),        Target :: netconf_db(),        Source :: netconf_db(),        Timeout :: timeout(),        Result :: ok | {error,error_reason()}. -%% @doc Copy configuration data. -%% -%% Which source and target options that can be issued depends on the -%% capabilities supported by the server. I.e. `:candidate' and/or -%% `:startup' are required. -%% -%% @end -%%----------------------------------------------------------------------  copy_config(Client, Target, Source, Timeout) ->      call(Client,{send_rpc_op, copy_config, [Target, Source], Timeout}).  %%---------------------------------------------------------------------- -%% @spec action(Client, Action) -> Result -%% @equiv action(Client, Action, infinity) +%% Execute an action. +-spec action(Client, Action) -> Result when +      Client :: client(), +      Action :: simple_xml(), +      Result :: ok | {ok,[simple_xml()]} | {error,error_reason()}.  action(Client,Action) ->      action(Client,Action,?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec action(Client, Action, Timeout) -> Result when        Client :: client(),        Action :: simple_xml(),        Timeout :: timeout(),        Result :: ok | {ok,[simple_xml()]} | {error,error_reason()}. -%% @doc Execute an action. If the return type is void, <c>ok</c> will -%%      be returned instead of <c>{ok,[simple_xml()]}</c>. -%% -%% @end -%%----------------------------------------------------------------------  action(Client,Action,Timeout) ->      call(Client,{send_rpc_op, action, [Action], Timeout}).  %%---------------------------------------------------------------------- +%% Send a 'create-subscription' request +%% See RFC5277, NETCONF Event Notifications +-spec create_subscription(Client) -> Result when +      Client :: client(), +      Result :: ok | {error,error_reason()}.  create_subscription(Client) ->      create_subscription(Client,?DEFAULT_STREAM,?DEFAULT_TIMEOUT). +-spec create_subscription(Client, Stream | Filter | Timeout) -> Result when +      Client :: client(), +      Stream :: stream_name(), +      Filter :: simple_xml() | [simple_xml()], +      Timeout :: timeout(), +      Result :: ok | {error,error_reason()}.  create_subscription(Client,Timeout)    when ?is_timeout(Timeout) ->      create_subscription(Client,?DEFAULT_STREAM,Timeout); @@ -876,6 +740,22 @@ create_subscription(Client,Stream,Filter,Timeout)  		 [Stream,Filter,undefined,undefined],  		 Timeout}). +-spec create_subscription(Client, Stream, StartTime, StopTime, Timeout) -> +				 Result when +      Client :: client(), +      Stream :: stream_name(), +      StartTime :: xs_datetime(), +      StopTime :: xs_datetime(), +      Timeout :: timeout(), +      Result :: ok | {error,error_reason()}; +                         (Client, Stream, Filter,StartTime, StopTime) -> +				 Result when +      Client :: client(), +      Stream :: stream_name(), +      Filter :: simple_xml() | [simple_xml()], +      StartTime :: xs_datetime(), +      StopTime :: xs_datetime(), +      Result :: ok | {error,error_reason()}.  create_subscription(Client,Stream,StartTime,StopTime,Timeout)    when ?is_string(Stream) andalso         ?is_string(StartTime) andalso @@ -891,7 +771,6 @@ create_subscription(Client,Stream,Filter,StartTime,StopTime)         ?is_string(StopTime) ->      create_subscription(Client,Stream,Filter,StartTime,StopTime,?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec create_subscription(Client, Stream, Filter,StartTime, StopTime, Timeout) ->  				 Result when        Client :: client(), @@ -901,168 +780,75 @@ create_subscription(Client,Stream,Filter,StartTime,StopTime)        StopTime :: xs_datetime(),        Timeout :: timeout(),        Result :: ok | {error,error_reason()}. -%% @doc Create a subscription for event notifications. -%% -%% This function sets up a subscription for netconf event -%% notifications of the given stream type, matching the given -%% filter. The calling process will receive notifications as messages -%% of type `notification()'. -%% -%% <dl> -%%   <dt>Stream:</dt> -%%   <dd> An optional parameter that indicates which stream of events -%%   is of interest.  If not present, events in the default NETCONF -%%   stream will be sent.</dd> -%% -%%   <dt>Filter:</dt> -%%   <dd>An optional parameter that indicates which subset of all -%%   possible events is of interest.  The format of this parameter is -%%   the same as that of the filter parameter in the NETCONF protocol -%%   operations.  If not present, all events not precluded by other -%%   parameters will be sent.</dd> -%% -%%   <dt>StartTime:</dt> -%%   <dd>An optional parameter used to trigger the replay feature and -%%   indicate that the replay should start at the time specified.  If -%%   `StartTime' is not present, this is not a replay subscription. -%%   It is not valid to specify start times that are later than the -%%   current time.  If the `StartTime' specified is earlier than the -%%   log can support, the replay will begin with the earliest -%%   available notification.  This parameter is of type dateTime and -%%   compliant to [RFC3339].  Implementations must support time -%%   zones.</dd> -%% -%%   <dt>StopTime:</dt> -%%   <dd>An optional parameter used with the optional replay feature -%%   to indicate the newest notifications of interest.  If `StopTime' -%%   is not present, the notifications will continue until the -%%   subscription is terminated.  Must be used with and be later than -%%   `StartTime'.  Values of `StopTime' in the future are valid.  This -%%   parameter is of type dateTime and compliant to [RFC3339]. -%%   Implementations must support time zones.</dd> -%% </dl> -%% -%% See RFC5277 for further details about the event notification -%% mechanism. -%% -%% @end -%%----------------------------------------------------------------------  create_subscription(Client,Stream,Filter,StartTime,StopTime,Timeout) ->      call(Client,{send_rpc_op,{create_subscription, self()},  		 [Stream,Filter,StartTime,StopTime],  		 Timeout}).  %%---------------------------------------------------------------------- -%% @spec get_event_streams(Client, Timeout) -> Result -%% @equiv get_event_streams(Client, [], Timeout) +%% Send a request to get the given event streams +%% See RFC5277, NETCONF Event Notifications +-spec get_event_streams(Client) +		       -> Result when +      Client :: client(), +      Result :: {ok,streams()} | {error,error_reason()}. +get_event_streams(Client) -> +    get_event_streams(Client,[],?DEFAULT_TIMEOUT). + +-spec get_event_streams(Client, Timeout) +		       -> Result when +      Client :: client(), +      Timeout :: timeout(), +      Result :: {ok,streams()} | {error,error_reason()}; +                       (Client, Streams) -> Result when +      Client :: client(), +      Streams :: [stream_name()], +      Result :: {ok,streams()} | {error,error_reason()}.  get_event_streams(Client,Timeout) when is_integer(Timeout); Timeout==infinity ->      get_event_streams(Client,[],Timeout); - -%%---------------------------------------------------------------------- -%% @spec get_event_streams(Client, Streams) -> Result -%% @equiv get_event_streams(Client, Streams, infinity)  get_event_streams(Client,Streams) when is_list(Streams) ->      get_event_streams(Client,Streams,?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec get_event_streams(Client, Streams, Timeout)  		       -> Result when        Client :: client(),        Streams :: [stream_name()],        Timeout :: timeout(),        Result :: {ok,streams()} | {error,error_reason()}. -%% @doc Send a request to get the given event streams. -%% -%% `Streams' is a list of stream names. The following filter will -%% be sent to the netconf server in a `get' request: -%% -%% ``` -%% <netconf xmlns="urn:ietf:params:xml:ns:netmod:notification"> -%%   <streams> -%%     <stream> -%%       <name>StreamName1</name> -%%     </stream> -%%     <stream> -%%       <name>StreamName2</name> -%%     </stream> -%%     ... -%%   </streams> -%% </netconf> -%% ''' -%% -%% If `Streams' is an empty list, ALL streams will be requested -%% by sending the following filter: -%% -%% ``` -%% <netconf xmlns="urn:ietf:params:xml:ns:netmod:notification"> -%%   <streams/> -%% </netconf> -%% ''' -%% -%% If more complex filtering is needed, a use {@link get/2} or {@link -%% get/3} and specify the exact filter according to XML Schema for -%% Event Notifications found in RFC5277. -%% -%% @end -%%----------------------------------------------------------------------  get_event_streams(Client,Streams,Timeout) ->      call(Client,{get_event_streams,Streams,Timeout}).  %%---------------------------------------------------------------------- -%% @spec close_session(Client) -> Result -%% @equiv close_session(Client, infinity) +%% Send a 'close-session' request +-spec close_session(Client) -> Result when +      Client :: client(), +      Result :: ok | {error,error_reason()}.  close_session(Client) ->      close_session(Client, ?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec close_session(Client, Timeout) -> Result when        Client :: client(),        Timeout :: timeout(),        Result :: ok | {error,error_reason()}. -%% @doc Request graceful termination of the session associated with the client. -%% -%% When a netconf server receives a `close-session' request, it -%% will gracefully close the session.  The server will release any -%% locks and resources associated with the session and gracefully -%% close any associated connections.  Any NETCONF requests received -%% after a `close-session' request will be ignored. -%% -%% @end -%%----------------------------------------------------------------------  close_session(Client, Timeout) ->      call(Client,{send_rpc_op, close_session, [], Timeout}, true).  %%---------------------------------------------------------------------- -%% @spec kill_session(Client, SessionId) -> Result -%% @equiv kill_session(Client, SessionId, infinity) +%% Send a 'kill-session' request +-spec kill_session(Client, SessionId) -> Result when +      Client :: client(), +      SessionId :: pos_integer(), +      Result :: ok | {error,error_reason()}.  kill_session(Client, SessionId) ->      kill_session(Client, SessionId, ?DEFAULT_TIMEOUT). -%%----------------------------------------------------------------------  -spec kill_session(Client, SessionId, Timeout) -> Result when        Client :: client(),        SessionId :: pos_integer(),        Timeout :: timeout(),        Result :: ok | {error,error_reason()}. -%% @doc Force termination of the session associated with the supplied -%% session id. -%% -%% The server side shall abort any operations currently in process, -%% release any locks and resources associated with the session, and -%% close any associated connections. -%% -%% Only if the server is in the confirmed commit phase, the -%% configuration will be restored to its state before entering the -%% confirmed commit phase. Otherwise, no configuration roll back will -%% be performed. -%% -%% If the given `SessionId' is equal to the current session id, -%% an error will be returned. -%% -%% @end -%% ----------------------------------------------------------------------  kill_session(Client, SessionId, Timeout) ->      call(Client,{send_rpc_op, kill_session, [SessionId], Timeout}). @@ -1071,24 +857,35 @@ kill_session(Client, SessionId, Timeout) ->  %% Callback functions  %%---------------------------------------------------------------------- -%% @private +init(_KeyOrName,{CM,{Host,Port}},Options) -> +    case ssh_channel(#connection{reference=CM,host=Host,port=Port},Options) of +        {ok,Connection} -> +	    {ok, CM, #state{connection = Connection}}; +	{error,Reason}-> +	    {error,Reason} +    end; +init(_KeyOrName,{_Host,_Port},Options) when Options#options.type==connection -> +    case ssh_connect(Options) of +        {ok, Connection} -> +	    ConnPid = Connection#connection.reference, +            {ok, ConnPid, #state{connection = Connection}}; +        Error -> +            Error +    end;  init(_KeyOrName,{_Host,_Port},Options) ->      case ssh_open(Options) of  	{ok, Connection} -> -	    log(Connection,open),  	    {ConnPid,_} = Connection#connection.reference,  	    {ok, ConnPid, #state{connection = Connection}};  	{error,Reason}->  	    {error,Reason}      end. -%% @private +  terminate(_, #state{connection=Connection}) ->      ssh_close(Connection), -    log(Connection,close),      ok. -%% @private  handle_msg({hello, Options, Timeout}, From,  	   #state{connection=Connection,hello_status=HelloStatus} = State) ->      case do_send(Connection, client_hello(Options)) of @@ -1107,6 +904,14 @@ handle_msg({hello, Options, Timeout}, From,  	Error ->  	    {stop, Error, State}      end; +handle_msg(get_ssh_connection, _From, #state{connection=Connection}=State) -> +    Reply = +        case Connection#connection.reference of +            {_,_} -> {error,not_an_ssh_connection}; +            CM -> {ok,{CM,{Connection#connection.host, +                           Connection#connection.port}}} +        end, +    {reply, Reply, State};  handle_msg(_, _From, #state{session_id=undefined} = State) ->      %% Hello is not yet excanged - this shall never happen      {reply,{error,waiting_for_hello},State}; @@ -1136,7 +941,6 @@ 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); @@ -1172,7 +976,6 @@ handle_msg({Ref,timeout},#state{pending=Pending} = State) ->      %% the implementation before this patch      {R,State#state{pending=Pending1, no_end_tag_buff= <<>>, buff= <<>>}}. -%% @private  %% Called by ct_util_server to close registered connections before terminate.  close(Client) ->      case get_handle(Client) of @@ -1243,15 +1046,18 @@ get_handle(Client) ->  	    Error      end. +check_options(OptList,Options) -> +    check_options(OptList,undefined,undefined,Options). +  check_options([], undefined, _Port, _Options) ->      {error, no_host_address};  check_options([], _Host, undefined, _Options) ->      {error, no_port};  check_options([], Host, Port, Options) ->      {Host,Port,Options}; -check_options([{ssh, Host}|T], _, Port, #options{} = Options) -> +check_options([{ssh, Host}|T], _, Port, Options) ->      check_options(T, Host, Port, Options#options{host=Host}); -check_options([{port,Port}|T], Host, _, #options{} = Options) -> +check_options([{port,Port}|T], Host, _, Options) ->      check_options(T, Host, Port, Options#options{port=Port});  check_options([{timeout, Timeout}|T], Host, Port, Options)    when is_integer(Timeout); Timeout==infinity -> @@ -1262,6 +1068,15 @@ check_options([Opt|T], Host, Port, #options{ssh=SshOpts}=Options) ->      %% Option verified by ssh      check_options(T, Host, Port, Options#options{ssh=[Opt|SshOpts]}). +check_session_options([],Options) -> +    {ok,Options}; +check_session_options([{timeout, Timeout}|T], Options) +  when is_integer(Timeout); Timeout==infinity -> +    check_session_options(T, Options#options{timeout = Timeout}); +check_session_options([Opt|_T], _Options) -> +    {error, {invalid_option, Opt}}. + +  %%%-----------------------------------------------------------------  set_request_timer(infinity) ->      {undefined,undefined}; @@ -1356,7 +1171,6 @@ do_send_rpc(Connection, MsgId, SimpleXml) ->  do_send(Connection, SimpleXml) ->      Xml=to_xml_doc(SimpleXml), -    log(Connection,send,Xml),      ssh_send(Connection, Xml).  to_xml_doc(Simple) -> @@ -1766,9 +1580,14 @@ decode_streams([]) ->  log(Connection,Action) ->      log(Connection,Action,<<>>). -log(#connection{host=Host,port=Port,name=Name},Action,Data) -> +log(#connection{reference=Ref,host=Host,port=Port,name=Name},Action,Data) -> +    Address = +        case Ref of +            {_,Ch} -> {Host,Port,Ch}; +            _ -> {Host,Port} +        end,      error_logger:info_report(#conn_log{client=self(), -				       address={Host,Port}, +				       address=Address,  				       name=Name,  				       action=Action,  				       module=?MODULE}, @@ -1776,7 +1595,6 @@ 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 @@ -1915,42 +1733,84 @@ get_tag([]) ->  %%%-----------------------------------------------------------------  %%% SSH stuff - -ssh_open(#options{host=Host,timeout=Timeout,port=Port,ssh=SshOpts,name=Name}) -> +ssh_connect(#options{host=Host,timeout=Timeout,port=Port, +                     ssh=SshOpts,name=Name,type=Type}) ->      case ssh:connect(Host, Port,  		     [{user_interaction,false}, -		      {silently_accept_hosts, true}|SshOpts]) of +		      {silently_accept_hosts, true}|SshOpts], +                     Timeout) of  	{ok,CM} -> -	    case ssh_connection:session_channel(CM, Timeout) of -		{ok,Ch} -> -		    case ssh_connection:subsystem(CM, Ch, "netconf", Timeout) of -			success -> -			    {ok, #connection{reference = {CM,Ch}, -					     host = Host, -					     port = Port, -					     name = Name}}; -			failure -> -			    ssh:close(CM), -			    {error,{ssh,could_not_execute_netconf_subsystem}}; -			{error,timeout} -> -			    {error,{ssh,could_not_execute_netconf_subsystem,timeout}} -		    end; -		{error, Reason} -> -		    ssh:close(CM), -		    {error,{ssh,could_not_open_channel,Reason}} -	    end; +            Connection = #connection{reference = CM, +                                     host = Host, +                                     port = Port, +                                     name = Name, +                                     type = Type}, +            log(Connection,connect), +            {ok,Connection};  	{error,Reason} ->  	    {error,{ssh,could_not_connect_to_server,Reason}}      end. -ssh_send(#connection{reference = {CM,Ch}}, Data) -> +ssh_channel(#connection{reference=CM}=Connection0, +            #options{timeout=Timeout,name=Name,type=Type}) -> +    case ssh_connection:session_channel(CM, Timeout) of +        {ok,Ch} -> +            case ssh_connection:subsystem(CM, Ch, "netconf", Timeout) of +                success -> +                    Connection = Connection0#connection{reference = {CM,Ch}, +                                                       name = Name, +                                                       type = Type}, +                    log(Connection,open), +                    {ok, Connection}; +                failure -> +                    ssh_connection:close(CM,Ch), +                    {error,{ssh,could_not_execute_netconf_subsystem}}; +                {error,timeout} -> +                    ssh_connection:close(CM,Ch), +                    {error,{ssh,could_not_execute_netconf_subsystem,timeout}} +            end; +        {error, Reason} -> +            {error,{ssh,could_not_open_channel,Reason}} +    end. + + +ssh_open(Options) -> +    case ssh_connect(Options) of +        {ok,Connection} -> +            case ssh_channel(Connection,Options) of +                {ok,_} = Ok -> +                    Ok; +                Error -> +                    ssh_close(Connection), +                    Error +            end; +        Error -> +            Error +    end. + +ssh_send(#connection{reference = {CM,Ch}}=Connection, Data) ->      case ssh_connection:send(CM, Ch, Data) of -	ok -> ok; -	{error,Reason} -> {error,{ssh,failed_to_send_data,Reason}} +	ok -> +            log(Connection,send,Data), +            ok; +	{error,Reason} -> +            {error,{ssh,failed_to_send_data,Reason}}      end. -ssh_close(#connection{reference = {CM,_Ch}}) -> -    ssh:close(CM). +ssh_close(Connection=#connection{reference = {CM,Ch}, type = Type}) -> +    _ = ssh_connection:close(CM,Ch), +    log(Connection,close), +    case Type of +        connection_and_channel -> +            ssh_close(Connection#connection{reference = CM}); +        _ -> +            ok +    end, +    ok; +ssh_close(Connection=#connection{reference = CM}) -> +    _ = ssh:close(CM), +    log(Connection,disconnect), +    ok.  %%---------------------------------------------------------------------- diff --git a/lib/common_test/src/ct_property_test.erl b/lib/common_test/src/ct_property_test.erl index 12c3d726d3..94ccb59af9 100644 --- a/lib/common_test/src/ct_property_test.erl +++ b/lib/common_test/src/ct_property_test.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 2003-2016. All Rights Reserved. +%% Copyright Ericsson AB 2003-2017. All Rights Reserved.  %%   %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -134,7 +134,7 @@ mk_ct_return(Other, Tool) ->      try lists:last(hd(Tool:counterexample()))      of  	{set,{var,_},{call,M,F,Args}} -> -	    {fail, io_lib:format("~p:~p/~p returned bad result",[M,F,length(Args)])} +	    {fail, io_lib:format("~p:~tp/~p returned bad result",[M,F,length(Args)])}      catch  	_:_ ->  	    {fail, Other} @@ -174,7 +174,7 @@ compile_tests(Path, ToolModule) ->      BeamFiles = [F || F<-FileNames,  		      filename:extension(F) == ".beam"],      _ = [file:delete(F) || F<-BeamFiles], -    ct:pal("Compiling in ~p:~n  Deleted ~p~n  MacroDefs=~p",[Path,BeamFiles,MacroDefs]), +    ct:pal("Compiling in ~tp:~n  Deleted ~p~n  MacroDefs=~p",[Path,BeamFiles,MacroDefs]),      Result = make:all([load|MacroDefs]),      ok = file:set_cwd(Cwd),      Result. diff --git a/lib/common_test/src/ct_release_test.erl b/lib/common_test/src/ct_release_test.erl index d783f8d04e..551a7e06d7 100644 --- a/lib/common_test/src/ct_release_test.erl +++ b/lib/common_test/src/ct_release_test.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2014-2016. All Rights Reserved. +%% Copyright Ericsson AB 2014-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -132,7 +132,7 @@  %%-----------------------------------------------------------------  -define(testnode, 'ct_release_test-upgrade'). --define(exclude_apps, [hipe, typer, dialyzer]). % never include these apps +-define(exclude_apps, [hipe, dialyzer]). % never include these apps  %%-----------------------------------------------------------------  -record(ct_data, {from,to}). @@ -445,7 +445,7 @@ init_upgrade_test(Level) ->  	end,      case OldRel of  	false -> -	    ct:log("Release ~p is not available." +	    ct:log("Release ~tp is not available."  		   " Upgrade on '~p' level can not be tested.",  		   [FromVsn,Level]),  	    undefined; @@ -522,8 +522,8 @@ upgrade(Apps,Level,Callback,CreateDir,InstallDir,Config) ->  		    {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]), +		    ct:log("Upgrade from: OTP-~ts, ~tp",[FromVsn, FromAppsVsns]), +		    ct:log("Upgrade to: OTP-~ts, ~tp",[ToVsn, ToAppsVsns]),  		    do_upgrade(Callback, FromVsn, FromAppsVsns, ToRel,  			       ToAppsVsns, InstallDir)  	    end @@ -727,9 +727,9 @@ do_upgrade({Cb,InitState},FromVsn,FromAppsVsns,ToRel,ToAppsVsns,InstallDir) ->  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]), +    ct:log("Calling ~p:~tp/1",[Mod,Func]),      R = rpc:call(Node,Mod,Func,Args), -    ct:log("~p:~p/~w returned: ~p",[Mod,Func,length(Args),R]), +    ct:log("~p:~tp/~w returned: ~tp",[Mod,Func,length(Args),R]),      case R of  	{badrpc,Error} ->  	    throw({fail,{test_upgrade_callback,Mod,Func,Args,Error}}); @@ -860,10 +860,7 @@ copy_file(Src, Dest, Opts) ->      end.  write_file(FName, Conts) -> -    Enc = file:native_name_encoding(), -    {ok, Fd} = file:open(FName, [write]), -    ok = file:write(Fd, unicode:characters_to_binary(Conts,Enc,Enc)), -    ok = file:close(Fd). +    file:write_file(FName, unicode:characters_to_binary(Conts)).  %% Substitute all occurrences of %Var% for Val in the given scripts  subst_src_scripts(Scripts, SrcDir, DestDir, Vars, Opts) -> @@ -879,7 +876,7 @@ subst_src_script(Script, SrcDir, DestDir, Vars, Opts) ->  subst_file(Src, Dest, Vars, Opts) ->      {ok, Bin} = file:read_file(Src), -    Conts = binary_to_list(Bin), +    Conts = unicode:characters_to_list(Bin),      NConts = subst(Conts, Vars),      write_file(Dest, NConts),      case lists:member(preserve, Opts) of @@ -891,7 +888,7 @@ subst_file(Src, Dest, Vars, Opts) ->      end.  subst(Str, [{Var,Val}|Vars]) -> -    subst(re:replace(Str,"%"++Var++"%",Val,[{return,list}]),Vars); +    subst(re:replace(Str,"%"++Var++"%",Val,[{return,list},unicode]),Vars);  subst(Str, []) ->      Str. diff --git a/lib/common_test/src/ct_repeat.erl b/lib/common_test/src/ct_repeat.erl index dac596a135..c043c9846c 100644 --- a/lib/common_test/src/ct_repeat.erl +++ b/lib/common_test/src/ct_repeat.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2007-2016. All Rights Reserved. +%% Copyright Ericsson AB 2007-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -43,7 +43,7 @@ loop_test(If,Args) when is_list(Args) ->  	no_loop ->  	    false;  	E = {error,_} -> -	    io:format("Common Test error: ~p\n\n",[E]), +	    io:format("Common Test error: ~tp\n\n",[E]),  	    ok = file:set_cwd(Cwd),  	    E;  	{repeat,N} -> @@ -89,18 +89,18 @@ loop(If,Type,N,Data0,Data1,Args,TPid,AccResult) ->  	{'EXIT',Pid,Reason} ->  	    case Reason of  		{user_error,What} -> -		    io:format("\nTest run failed!\nReason: ~p\n\n\n", [What]), +		    io:format("\nTest run failed!\nReason: ~tp\n\n\n", [What]),  		    cancel(TPid),  		    {error,What};			  		_ ->  		    io:format("Test run crashed! This could be an internal error "  			      "- please report!\n\n" -			      "~p\n\n\n",[Reason]), +			      "~tp\n\n\n",[Reason]),  		    cancel(TPid),  		    {error,Reason}  	    end;  	{Pid,{error,Reason}} -> -	    io:format("\nTest run failed!\nReason: ~p\n\n\n",[Reason]), +	    io:format("\nTest run failed!\nReason: ~tp\n\n\n",[Reason]),  	    cancel(TPid),  	    {error,Reason};  	{Pid,Result} -> diff --git a/lib/common_test/src/ct_run.erl b/lib/common_test/src/ct_run.erl index 10e62e18b8..14f28f9ca3 100644 --- a/lib/common_test/src/ct_run.erl +++ b/lib/common_test/src/ct_run.erl @@ -121,13 +121,13 @@ script_start() ->  		%% used for purpose of testing the run_test interface  		io:format(user, "~n-------------------- START ARGS "  			  "--------------------~n", []), -		io:format(user, "--- Init args:~n~p~n", [FlagFilter(Init)]), -		io:format(user, "--- CT args:~n~p~n", [FlagFilter(CtArgs)]), +		io:format(user, "--- Init args:~n~tp~n", [FlagFilter(Init)]), +		io:format(user, "--- CT args:~n~tp~n", [FlagFilter(CtArgs)]),  		EnvArgs = opts2args(EnvStartOpts), -		io:format(user, "--- Env opts -> args:~n~p~n   =>~n~p~n", +		io:format(user, "--- Env opts -> args:~n~tp~n   =>~n~tp~n",  			  [EnvStartOpts,EnvArgs]),  		Merged = merge_arguments(CtArgs ++ EnvArgs), -		io:format(user, "--- Merged args:~n~p~n", [FlagFilter(Merged)]), +		io:format(user, "--- Merged args:~n~tp~n", [FlagFilter(Merged)]),  		io:format(user, "-----------------------------------"  			  "-----------------~n~n", []),  		Merged; @@ -160,18 +160,18 @@ script_start(Args) ->  		{'EXIT',Pid,Reason} ->  		    case Reason of  			{user_error,What} -> -			    io:format("\nTest run failed!\nReason: ~p\n\n\n", +			    io:format("\nTest run failed!\nReason: ~tp\n\n\n",                                        [What]),  			    finish(Tracing, ?EXIT_STATUS_TEST_RUN_FAILED, Args);  			_ ->  			    io:format("Test run crashed! "                                        "This could be an internal error "  				      "- please report!\n\n" -				      "~p\n\n\n", [Reason]), +				      "~tp\n\n\n", [Reason]),  			    finish(Tracing, ?EXIT_STATUS_TEST_RUN_FAILED, Args)  		    end;  		{Pid,{error,Reason}} -> -		    io:format("\nTest run failed! Reason:\n~p\n\n\n",[Reason]), +		    io:format("\nTest run failed! Reason:\n~tp\n\n\n",[Reason]),  		    finish(Tracing, ?EXIT_STATUS_TEST_RUN_FAILED, Args);  		{Pid,Result} ->  		    io:nl(), @@ -219,7 +219,7 @@ analyze_test_result([], _) ->  analyze_test_result(interactive_mode, _) ->      interactive_mode;  analyze_test_result(Unknown, _) -> -    io:format("\nTest run failed! Reason:\n~p\n\n\n",[Unknown]), +    io:format("\nTest run failed! Reason:\n~tp\n\n\n",[Unknown]),      ?EXIT_STATUS_TEST_RUN_FAILED.  finish(Tracing, ExitStatus, Args) -> @@ -363,6 +363,12 @@ script_start1(Parent, Args) ->  	_ ->  	    application:set_env(common_test, disable_log_cache, true)      end, +    %% log_cleanup - used by ct_logs +    KeepLogs = get_start_opt(keep_logs, +                             fun ct_logs:parse_keep_logs/1, +                             all, +                             Args), +    application:set_env(common_test, keep_logs, KeepLogs),      Opts = #opts{label = Label, profile = Profile,  		 vts = Vts, shell = Shell, @@ -430,13 +436,17 @@ script_start2(Opts = #opts{vts = undefined,  	    Specs1 = get_start_opt(join_specs, [Specs], Specs, Args),  	    %% using testspec as input for test  	    Relaxed = get_start_opt(allow_user_terms, true, false, Args), -	    case catch ct_testspec:collect_tests_from_file(Specs1, Relaxed) of -		{E,Reason} when E == error ; E == 'EXIT' -> +	    try ct_testspec:collect_tests_from_file(Specs1, Relaxed) of +                TestSpecData -> +		    execute_all_specs(TestSpecData, Opts, Args, []) +            catch +                throw:{error,Reason} ->  		    StackTrace = erlang:get_stacktrace(),  		    {error,{invalid_testspec,{Reason,StackTrace}}}; -		TestSpecData -> -		    execute_all_specs(TestSpecData, Opts, Args, []) -	    end; +                _:Reason -> +		    StackTrace = erlang:get_stacktrace(), +		    {error,{invalid_testspec,{Reason,StackTrace}}} +            end;  	[] ->  	    {error,no_testspec_specified};  	_ ->	    % no testspec used @@ -750,7 +760,7 @@ script_start4(#opts{label = Label, profile = Profile,      if Config == [] ->  	    ok;         true -> -	    io:format("\nInstalling: ~p\n\n", [Config]) +	    io:format("\nInstalling: ~tp\n\n", [Config])      end,      case install([{config,Config},{event_handler,EvHandlers},  		  {ct_hooks, CTHooks}, @@ -899,9 +909,9 @@ install(Opts, LogDir) ->      case whereis(ct_util_server) of  	undefined ->  	    VarFile = variables_file_name(LogDir), -	    case file:open(VarFile, [write]) of +	    case file:open(VarFile, [write, {encoding,utf8}]) of  		{ok,Fd} -> -		    _ = [io:format(Fd, "~p.\n", [Opt]) || Opt <- ConfOpts], +		    _ = [io:format(Fd, "~tp.\n", [Opt]) || Opt <- ConfOpts],  		    ok = file:close(Fd);  		{error,Reason} ->  		    io:format("CT failed to install configuration data. Please " @@ -970,6 +980,12 @@ run_test1(StartOpts) when is_list(StartOpts) ->  	    stop_trace(Tracing),  	    exit(Res);  	RefreshDir -> +            %% log_cleanup - used by ct_logs +            KeepLogs = get_start_opt(keep_logs, +                                     fun ct_logs:parse_keep_logs/1, +                                     all, +                                     StartOpts), +            application:set_env(common_test, keep_logs, KeepLogs),  	    ok = refresh_logs(?abs(RefreshDir)),  	    exit(done)      end. @@ -1131,6 +1147,12 @@ run_test2(StartOpts) ->  	DisableCacheBool ->  	    application:set_env(common_test, disable_log_cache, DisableCacheBool)      end, +    %% log_cleanup - used by ct_logs +    KeepLogs = get_start_opt(keep_logs, +                             fun ct_logs:parse_keep_logs/1, +                             all, +                             StartOpts), +    application:set_env(common_test, keep_logs, KeepLogs),      %% stepped execution      Step = get_start_opt(step, value, StartOpts), @@ -1180,12 +1202,16 @@ run_spec_file(Relaxed,  	     end,      AbsSpecs = lists:map(fun(SF) -> ?abs(SF) end, Specs1),      AbsSpecs1 = get_start_opt(join_specs, [AbsSpecs], AbsSpecs, StartOpts), -    case catch ct_testspec:collect_tests_from_file(AbsSpecs1, Relaxed) of -	{Error,CTReason} when Error == error ; Error == 'EXIT' -> -	    StackTrace = erlang:get_stacktrace(), -	    exit({error,{invalid_testspec,{CTReason,StackTrace}}}); +    try ct_testspec:collect_tests_from_file(AbsSpecs1, Relaxed) of  	TestSpecData ->  	    run_all_specs(TestSpecData, Opts, StartOpts, []) +    catch +	throw:{error,CTReason} -> +	    StackTrace = erlang:get_stacktrace(), +	    exit({error,{invalid_testspec,{CTReason,StackTrace}}}); +	_:CTReason -> +	    StackTrace = erlang:get_stacktrace(), +	    exit({error,{invalid_testspec,{CTReason,StackTrace}}})      end.  run_all_specs([], _, _, TotResult) -> @@ -1802,10 +1828,10 @@ compile_and_run(Tests, Skip, Opts, Args) ->  	    case lists:member(all, Conns) of  		true ->  		    Conns1 = ct_util:override_silence_all_connections(), -		    ct_logs:log("Silent connections", "~p", [Conns1]); +		    ct_logs:log("Silent connections", "~tp", [Conns1]);  		false ->  		    ct_util:override_silence_connections(Conns), -		    ct_logs:log("Silent connections", "~p", [Conns]) +		    ct_logs:log("Silent connections", "~tp", [Conns])  	    end      end,      log_ts_names(Opts#opts.testspec_files), @@ -1898,7 +1924,7 @@ possibly_spawn(true, Tests, Skip, Opts) ->  		TestRunPid = spawn_link(TestRun),  		receive  		    {'EXIT',TestRunPid,{ok,TestResult}} -> -			io:format(user, "~nCommon Test returned ~p~n~n", +			io:format(user, "~nCommon Test returned ~tp~n~n",  				  [TestResult]);  		    {'EXIT',TestRunPid,Error} ->  			exit(Error)				 @@ -1917,7 +1943,7 @@ auto_compile(TestSuites) ->  	case application:get_env(common_test, include) of  	    {ok,UserInclDirs} when length(UserInclDirs) > 0 ->  		io:format("Including the following directories:~n"), -		[begin io:format("~p~n",[UserInclDir]), {i,UserInclDir} end || +		[begin io:format("~tp~n",[UserInclDir]), {i,UserInclDir} end ||  		 UserInclDir <- UserInclDirs];  	    _ ->  		[] @@ -2258,7 +2284,7 @@ do_run_test(Tests, Skip, Opts0) ->  	    NoOfSuites = length(Suites1),  	    ct_util:warn_duplicates(Suites1),  	    {ok,Cwd} = file:get_cwd(), -	    io:format("~nCWD set to: ~p~n", [Cwd]), +	    io:format("~nCWD set to: ~tp~n", [Cwd]),  	    if NoOfCases == unknown ->  		    io:format("~nTEST INFO: ~w test(s), ~w suite(s)~n~n",  			      [NoOfTests,NoOfSuites]), @@ -2328,7 +2354,7 @@ do_run_test(Tests, Skip, Opts0) ->  	    case ct_util:get_testdata(severe_error) of  		undefined -> ok;  		SevereError -> -		    ct_logs:log("SEVERE ERROR", "~p\n", [SevereError]), +		    ct_logs:log("SEVERE ERROR", "~tp\n", [SevereError]),  		    exit(SevereError)  	    end, @@ -2399,7 +2425,7 @@ start_cover(Opts=#opts{coverspec=CovData,cover_stop=CovStop},LogDir) ->      if (CovNodes /= []) and (CovNodes /= undefined) ->  	    ct_logs:log("COVER INFO",  			"Nodes included in cover " -			"session: ~w", +			"session: ~tw",  			[CovNodes]),  	    cover:start(CovNodes);         true -> @@ -2413,7 +2439,7 @@ start_cover(Opts=#opts{coverspec=CovData,cover_stop=CovStop},LogDir) ->  		  {error,Reason} ->  		      ct_logs:log("COVER INFO",  				  "Importing cover data from: ~ts fails! " -				  "Reason: ~p", [Imp,Reason]) +				  "Reason: ~tp", [Imp,Reason])  	      end        end, CovImport),      {TsCoverInfo,Opts}. @@ -2747,7 +2773,7 @@ run_make(Targets, TestDir0, Mod, UserInclude) ->  		{up_to_date,_} ->  		    ok;  		{'EXIT',Reason} -> -		    io:format("{error,{make_crashed,~p}\n", [Reason]), +		    io:format("{error,{make_crashed,~tp}\n", [Reason]),  		    {error,{make_crashed,TestDir,Reason}};  		{error,ModInfo} ->  		    io:format("{error,make_failed}\n", []), @@ -2756,7 +2782,7 @@ run_make(Targets, TestDir0, Mod, UserInclude) ->  		    {error,{make_failed,Bad}}  	    end;  	{error,_} -> -	    io:format("{error,{invalid_directory,~p}}\n", [TestDir0]), +	    io:format("{error,{invalid_directory,~tp}}\n", [TestDir0]),  	    {error,{invalid_directory,TestDir0}}      end. @@ -2806,7 +2832,7 @@ maybe_interpret2(Suite, Cases, StepOpts) ->  	       _ -> ok  	   catch  	       _:_Error -> -		   io:format(user, "Invalid breakpoint: ~w:~w/1~n", +		   io:format(user, "Invalid breakpoint: ~w:~tw/1~n",  			     [Suite,Case])  	   end   	 end || Case <- Cases, is_atom(Case)], @@ -2937,7 +2963,7 @@ ct_hooks_args2opts([],Acc) ->  parse_cth_args(String) ->      try -	true = io_lib:printable_list(String), +	true = io_lib:printable_unicode_list(String),  	{ok,Toks,_} = erl_scan:string(String++"."),  	{ok, Args} = erl_parse:parse_term(Toks),  	Args @@ -3016,7 +3042,7 @@ rel_to_abs(CtArgs) ->  	 _ = if Dir /= Abs ->  		 _ = code:del_path(Dir),  		 _ = code:del_path(Abs),		  -		 io:format(user, "Converting ~p to ~p and re-inserting " +		 io:format(user, "Converting ~tp to ~tp and re-inserting "  			   "with add_pathz/1~n",  			   [Dir, Abs]);  	    true -> @@ -3030,7 +3056,7 @@ rel_to_abs(CtArgs) ->  	 _ = if Dir /= Abs ->  		 _ = code:del_path(Dir),  		 _ = code:del_path(Abs),		  -		 io:format(user, "Converting ~p to ~p and re-inserting " +		 io:format(user, "Converting ~tp to ~tp and re-inserting "  			   "with add_patha/1~n",  			   [Dir, Abs]);  	    true -> @@ -3100,7 +3126,7 @@ opts2args(EnvStartOpts) ->  		     ({group,G}) when is_atom(G) ->  			  [{group,[atom_to_list(G)]}];  		     ({group,Gs}) when is_list(Gs) -> -			  LOfGStrs = [lists:flatten(io_lib:format("~w",[G])) || +			  LOfGStrs = [lists:flatten(io_lib:format("~tw",[G])) ||  					 G <- Gs],  			  [{group,LOfGStrs}];  		     ({testcase,Case}) when is_atom(Case) -> @@ -3152,10 +3178,10 @@ opts2args(EnvStartOpts) ->  		     ({event_handler,EHs}) when is_list(EHs) ->  			  [{event_handler,[atom_to_list(EH) || EH <- EHs]}];  		     ({event_handler,{EH,Arg}}) when is_atom(EH) -> -			  ArgStr = lists:flatten(io_lib:format("~p", [Arg])), +			  ArgStr = lists:flatten(io_lib:format("~tp", [Arg])),  			  [{event_handler_init,[atom_to_list(EH),ArgStr]}];  		     ({event_handler,{EHs,Arg}}) when is_list(EHs) -> -			  ArgStr = lists:flatten(io_lib:format("~p", [Arg])), +			  ArgStr = lists:flatten(io_lib:format("~tp", [Arg])),  			  Strs = lists:flatmap(fun(EH) ->  						       [atom_to_list(EH),  							ArgStr,"and"] @@ -3186,25 +3212,25 @@ opts2args(EnvStartOpts) ->  		     ({ct_hooks,[]}) ->  			  [];  		     ({ct_hooks,CTHs}) when is_list(CTHs) -> -			  io:format(user,"ct_hooks: ~p",[CTHs]), +			  io:format(user,"ct_hooks: ~tp",[CTHs]),  			  Strs = lists:flatmap(  				   fun({CTH,Arg,Prio}) ->  					   [atom_to_list(CTH),  					    lists:flatten( -					      io_lib:format("~p",[Arg])), +					      io_lib:format("~tp",[Arg])),  					    lists:flatten( -					      io_lib:format("~p",[Prio])), +					      io_lib:format("~tp",[Prio])),  					    "and"];  				       ({CTH,Arg}) ->  					   [atom_to_list(CTH),  					    lists:flatten( -					      io_lib:format("~p",[Arg])), +					      io_lib:format("~tp",[Arg])),  					    "and"];  				      (CTH) when is_atom(CTH) ->  					   [atom_to_list(CTH),"and"]  				   end,CTHs),  			  [_LastAnd|StrsR] = lists:reverse(Strs), -			  io:format(user,"return: ~p",[lists:reverse(StrsR)]), +			  io:format(user,"return: ~tp",[lists:reverse(StrsR)]),  			  [{ct_hooks,lists:reverse(StrsR)}];  		     ({Opt,As=[A|_]}) when is_atom(A) ->  			  [{Opt,[atom_to_list(Atom) || Atom <- As]}]; @@ -3286,7 +3312,7 @@ start_trace(Args) ->  			ok ->  			    true;  			{_,Error} -> -			    io:format("Warning! Tracing not started. Reason: ~p~n~n", +			    io:format("Warning! Tracing not started. Reason: ~tp~n~n",  				      [Error]),  			    false  		    end; diff --git a/lib/common_test/src/ct_slave.erl b/lib/common_test/src/ct_slave.erl index 571958ca03..4188bd7c3b 100644 --- a/lib/common_test/src/ct_slave.erl +++ b/lib/common_test/src/ct_slave.erl @@ -309,7 +309,12 @@ is_started(ENode) ->  % make a Erlang node name from name and hostname  enodename(Host, Node) -> -    list_to_atom(atom_to_list(Node)++"@"++atom_to_list(Host)). +    case lists:member($@, atom_to_list(Node)) of +        true -> +            Node; +        false -> +            list_to_atom(atom_to_list(Node)++"@"++atom_to_list(Host)) +    end.  % performs actual start of the "slave" node  do_start(Host, Node, Options) -> diff --git a/lib/common_test/src/ct_ssh.erl b/lib/common_test/src/ct_ssh.erl index 6ab3bf036c..ca62357e1c 100644 --- a/lib/common_test/src/ct_ssh.erl +++ b/lib/common_test/src/ct_ssh.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2009-2016. All Rights Reserved. +%% Copyright Ericsson AB 2009-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -68,7 +68,8 @@   	 send_and_receive/3, send_and_receive/4, send_and_receive/5,  	 send_and_receive/6,   	 exec/2, exec/3, exec/4, - 	 subsystem/3, subsystem/4]). +         subsystem/3, subsystem/4, +         shell/2, shell/3]).  %% STFP Functions  -export([sftp_connect/1,  @@ -94,6 +95,7 @@  -record(state, {ssh_ref, conn_type, target}). +-type handle() :: pid().  %%%-----------------------------------------------------------------  %%%------------------------ SSH COMMANDS --------------------------- @@ -159,7 +161,7 @@ connect(KeyOrName, ExtraOpts) when is_list(ExtraOpts) ->  connect(KeyOrName, ConnType, ExtraOpts) ->      case ct:get_config(KeyOrName) of  	undefined -> -	    log(heading(connect,KeyOrName), "Failed: ~p\n", +	    log(heading(connect,KeyOrName), "Failed: ~tp\n",  		[{not_available,KeyOrName}]),  	    {error,{not_available,KeyOrName}};  	SSHData -> @@ -212,18 +214,18 @@ connect(KeyOrName, ConnType, ExtraOpts) ->  		end,  	    case {Addr,proplists:get_value(port, AllOpts1)} of  		{undefined,_} -> -		    log(heading(connect,KeyOrName), "Failed: ~p\n", +		    log(heading(connect,KeyOrName), "Failed: ~tp\n",  			[{not_available,{KeyOrName,ConnType1}}]),  		    {error,{not_available,{KeyOrName,ConnType1}}};  		{_,undefined} ->  		    try_log(heading(connect,KeyOrName),  -			    "Opening ~w connection to ~p:22\n", +			    "Opening ~w connection to ~tp:22\n",  			    [ConnType1,Addr]),  		    ct_gen_conn:start(KeyOrName, {ConnType1,Addr,22},   				      AllOpts1, ?MODULE);		      		{_,Port} ->  		    try_log(heading(connect,KeyOrName),  -			    "Opening ~w connection to ~p:~w\n", +			    "Opening ~w connection to ~tp:~w\n",  			    [ConnType1,Addr,Port]),  		    ct_gen_conn:start(KeyOrName, {ConnType1,Addr,Port},   				      AllOpts1, ?MODULE) @@ -490,6 +492,22 @@ subsystem(SSH, ChannelId, Subsystem, Timeout) ->      call(SSH, {subsystem,ChannelId,Subsystem,Timeout}). +-spec shell(SSH, ChannelId) -> Result when +      SSH :: handle() | ct:target_name(), +      ChannelId :: ssh:ssh_channel_id(), +      Result :: ok | {error,term()}. +shell(SSH, ChannelId) -> +    shell(SSH, ChannelId, ?DEFAULT_TIMEOUT). + +-spec shell(SSH, ChannelId, Timeout) -> Result when +      SSH :: handle() | ct:target_name(), +      ChannelId :: ssh:ssh_channel_id(), +      Timeout :: timeout(), +      Result :: ok | {error,term()}. +shell(SSH, ChannelId, Timeout) -> +    call(SSH, {shell,ChannelId,Timeout}). + +  %%%-----------------------------------------------------------------  %%%------------------------ SFTP COMMANDS -------------------------- @@ -977,7 +995,7 @@ init(KeyOrName, {ConnType,Addr,Port}, AllOpts) ->  	    SSHRef = element(2, Ok),  	    try_log(heading(init,KeyOrName),   		    "Opened ~w connection:\n" -		    "Host: ~p (~p)\nUser: ~p\nPassword: ~p\n", +		    "Host: ~tp (~p)\nUser: ~tp\nPassword: ~p\n",  		[ConnType,Addr,Port,User,lists:duplicate(length(Password),$*)]),  	    {ok,SSHRef,#state{ssh_ref=SSHRef, conn_type=ConnType,  			      target=KeyOrName}} @@ -1015,11 +1033,11 @@ handle_msg({exec,Chn,Command,TO}, State) ->  	end,      case Chn1 of  	{error,_} = ChnError -> -	    log(heading(exec,Target), "Opening channel failed: ~p", [ChnError]), +	    log(heading(exec,Target), "Opening channel failed: ~tp", [ChnError]),  	    {ChnError,State};  	_ ->  	    try_log(heading(exec,Target),  -		    "SSH Ref: ~p, Chn: ~p, Command: ~p, Timeout: ~p",  +		    "SSH Ref: ~p, Chn: ~p, Command: ~tp, Timeout: ~p",  		    [SSHRef,Chn1,Command,TO]),  	    case ssh_connection:exec(SSHRef, Chn1, Command, TO) of  		success -> @@ -1042,7 +1060,7 @@ handle_msg({send,Chn,Type,Data,TO}, State) ->      #state{ssh_ref=SSHRef, target=Target} = State,      try_log(heading(send,Target),   	    "SSH Ref: ~p, Chn: ~p, Type: ~p, Timeout: ~p~n" -	    "Data: ~p", [SSHRef,Chn,Type,TO,Data]), +	    "Data: ~tp", [SSHRef,Chn,Type,TO,Data]),      Result = ssh_connection:send(SSHRef, Chn, Type, Data, TO),      {Result,State}; @@ -1050,7 +1068,7 @@ handle_msg({send_and_receive,Chn,Type,Data,End,TO}, State) ->      #state{ssh_ref=SSHRef, target=Target} = State,         try_log(heading(send_and_receive,Target),   	    "SSH Ref: ~p, Chn: ~p, Type: ~p, Timeout: ~p~n" -	    "Data: ~p", [SSHRef,Chn,Type,TO,Data]), +	    "Data: ~tp", [SSHRef,Chn,Type,TO,Data]),      case ssh_connection:send(SSHRef, Chn, Type, Data, TO) of  	ok ->  	    Result = do_recv_response(SSHRef, Chn, [], End, TO), @@ -1062,160 +1080,168 @@ handle_msg({send_and_receive,Chn,Type,Data,End,TO}, State) ->  handle_msg({subsystem,Chn,Subsystem,TO}, State) ->      #state{ssh_ref=SSHRef, target=Target} = State,      try_log(heading(subsystem,Target),  -	    "SSH Ref: ~p, Chn: ~p, Subsys: ~p, Timeout: ~p",  +	    "SSH Ref: ~p, Chn: ~p, Subsys: ~tp, Timeout: ~p",  	    [SSHRef,Chn,Subsystem,TO]),      Result = ssh_connection:subsystem(SSHRef, Chn, Subsystem, TO),      {Result,State}; +handle_msg({shell,Chn,TO}, State) -> +    #state{ssh_ref=SSHRef, target=Target} = State, +    try_log(heading(shell,Target), +	    "SSH Ref: ~p, Chn: ~p, Timeout: ~p", +            [SSHRef,Chn,TO]), +    Result = ssh_connection:shell(SSHRef, Chn), +    {Result,State}; +  %% --- SFTP Commands ---  handle_msg({read_file,Srv,File}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:read_file(ref(Srv,SSHRef), File),S};  handle_msg({write_file,Srv,File,Iolist}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:write_file(ref(Srv,SSHRef), File, Iolist),S};  handle_msg({list_dir,Srv,Path}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:list_dir(ref(Srv,SSHRef), Path),S};  handle_msg({open,Srv,File,Mode}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:open(ref(Srv,SSHRef), File, Mode),S};  handle_msg({opendir,Srv,Path}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:opendir(ref(Srv,SSHRef), Path),S};  handle_msg({close,Srv,Handle}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:close(ref(Srv,SSHRef), Handle),S};  handle_msg({read,Srv,Handle,Len}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:read(ref(Srv,SSHRef), Handle, Len),S};  handle_msg({pread,Srv,Handle,Position,Length}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:pread(ref(Srv,SSHRef),Handle,Position,Length),S};  handle_msg({aread,Srv,Handle,Len}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:aread(ref(Srv,SSHRef), Handle, Len),S};  handle_msg({apread,Srv,Handle,Position,Length}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:apread(ref(Srv,SSHRef), Handle, Position, Length),S};  handle_msg({write,Srv,Handle,Data}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:write(ref(Srv,SSHRef), Handle, Data),S};  handle_msg({pwrite,Srv,Handle,Position,Data}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:pwrite(ref(Srv,SSHRef), Handle, Position, Data),S};  handle_msg({awrite,Srv,Handle,Data}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:awrite(ref(Srv,SSHRef), Handle, Data),S};  handle_msg({apwrite,Srv,Handle,Position,Data}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:apwrite(ref(Srv,SSHRef), Handle, Position, Data),S};  handle_msg({position,Srv,Handle,Location}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:position(ref(Srv,SSHRef), Handle, Location),S};  handle_msg({read_file_info,Srv,Name}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:read_file_info(ref(Srv,SSHRef), Name),S};  handle_msg({get_file_info,Srv,Handle}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:get_file_info(ref(Srv,SSHRef), Handle),S};  handle_msg({read_link_info,Srv,Name}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:read_link_info(ref(Srv,SSHRef), Name),S};  handle_msg({write_file_info,Srv,Name,Info}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:write_file_info(ref(Srv,SSHRef), Name, Info),S};  handle_msg({read_link,Srv,Name}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:read_link(ref(Srv,SSHRef), Name),S};  handle_msg({make_symlink,Srv,Name,Target}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:make_symlink(ref(Srv,SSHRef), Name, Target),S};  handle_msg({rename,Srv,OldName,NewName}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:rename(ref(Srv,SSHRef), OldName, NewName),S};  handle_msg({delete,Srv,Name}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:delete(ref(Srv,SSHRef), Name),S};  handle_msg({make_dir,Srv,Name}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:make_dir(ref(Srv,SSHRef), Name),S};  handle_msg({del_dir,Srv,Name}=Cmd, S=#state{ssh_ref=SSHRef}) ->      try_log(heading(sftp,S#state.target),  -	    "SSH Ref: ~p, Server: ~p~nCmd: ~p", +	    "SSH Ref: ~p, Server: ~p~nCmd: ~tp",  	    [SSHRef,ref(Srv,SSHRef),mod(Cmd)]),      {ssh_sftp:del_dir(ref(Srv,SSHRef), Name),S}. @@ -1259,7 +1285,7 @@ do_recv_response(SSH, Chn, Data, End, Timeout) ->  	{ssh_cm, SSH, {data,Chn,_,NewData}} ->  	    ssh_connection:adjust_window(SSH, Chn, size(NewData)), -	    debug("RECVD~n~p", [binary_to_list(NewData)]), +	    debug("RECVD~n~tp", [binary_to_list(NewData)]),  	    DataAcc = Data ++ binary_to_list(NewData),  	    if is_function(End) ->  		    case End(DataAcc) of @@ -1312,7 +1338,7 @@ do_recv_response(SSH, Chn, Data, End, Timeout) ->  	%%	    {ok,WCh};  	Other -> -	    debug("UNEXPECTED MESSAGE~n~p ~p~n~p", [SSH,Chn,Other]), +	    debug("UNEXPECTED MESSAGE~n~p ~p~n~tp", [SSH,Chn,Other]),  	    do_recv_response(SSH, Chn, Data, End, Timeout)      after Timeout -> @@ -1365,7 +1391,7 @@ mod(Cmd) ->  %%%-----------------------------------------------------------------  %%% 	    heading(Function, Ref) -> -    io_lib:format("ct_ssh:~w ~p",[Function,Ref]). +    io_lib:format("ct_ssh:~tw ~tp",[Function,Ref]).  %%%-----------------------------------------------------------------  %%%  diff --git a/lib/common_test/src/ct_telnet.erl b/lib/common_test/src/ct_telnet.erl index bff1112ab9..b50cddd492 100644 --- a/lib/common_test/src/ct_telnet.erl +++ b/lib/common_test/src/ct_telnet.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2003-2016. All Rights Reserved. +%% Copyright Ericsson AB 2003-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -251,7 +251,7 @@ open(KeyOrName,ConnType,TargetMod) ->  open(KeyOrName,ConnType,TargetMod,Extra) ->      case ct:get_config({KeyOrName,ConnType}) of  	undefined -> -	    log(undefined,open,"Failed: ~p",[{not_available,KeyOrName}]), +	    log(undefined,open,"Failed: ~tp",[{not_available,KeyOrName}]),  	    {error,{not_available,KeyOrName,ConnType}};  	Addr ->  	    Addr1 = @@ -273,7 +273,7 @@ open(KeyOrName,ConnType,TargetMod,Extra) ->  			end;  		    Bool -> Bool  		end, -	    log(undefined,open,"Connecting to ~p(~p)", +	    log(undefined,open,"Connecting to ~tp(~tp)",  		[KeyOrName,Addr1]),  	    Reconnect =  		case ct:get_config({telnet_settings,reconnection_attempts}) of @@ -672,7 +672,7 @@ set_telnet_defaults([{tcp_nodelay,NoDelay}|Ss],S) ->      set_telnet_defaults(Ss,S#state{tcp_nodelay=NoDelay});  set_telnet_defaults([Unknown|Ss],S) ->      force_log(S,error, -	      "Bad element in telnet_settings: ~p",[Unknown]), +	      "Bad element in telnet_settings: ~tp",[Unknown]),      set_telnet_defaults(Ss,S);  set_telnet_defaults([],S) ->      S. @@ -680,7 +680,7 @@ set_telnet_defaults([],S) ->  %% @hidden  handle_msg({cmd,Cmd,Opts},State) ->      start_gen_log(heading(cmd,State#state.name)), -    log(State,cmd,"Cmd: ~p",[Cmd]), +    log(State,cmd,"Cmd: ~tp",[Cmd]),      %% whatever is in the buffer from previous operations      %% will be ignored as we go ahead with this telnet cmd @@ -715,7 +715,7 @@ handle_msg({cmd,Cmd,Opts},State) ->  	case teln_cmd(State#state.teln_pid, Cmd, State#state.prx,  		      Newline, TO) of  	    {ok,Data,_PromptType,Rest} -> -		log(State,recv,"Return: ~p",[{ok,Data}]), +		log(State,recv,"Return: ~tp",[{ok,Data}]),  		{{ok,Data},Rest,true};  	    Error ->  		Retry = {retry,{Error, @@ -723,14 +723,14 @@ handle_msg({cmd,Cmd,Opts},State) ->  				 State#state.type},  				State#state.teln_pid,  				{cmd,Cmd,Opts}}}, -		log(State,recv,"Return: ~p",[Error]), +		log(State,recv,"Return: ~tp",[Error]),  		{Retry,[],false}  	end,      end_gen_log(),      {Return,State#state{buffer=NewBuffer,prompt=Prompt}};  handle_msg({send,Cmd,Opts},State) ->      start_gen_log(heading(send,State#state.name)), -    log(State,send,"Sending: ~p",[Cmd]), +    log(State,send,"Sending: ~tp",[Cmd]),      debug_cont_gen_log("Throwing Buffer:",[]),      debug_log_lines(State#state.buffer), @@ -762,12 +762,12 @@ handle_msg(get_data,State) ->      log(State,cmd,"Reading data...",[]),      {ok,Data,Buffer} = teln_get_all_data(State,State#state.buffer,[],[],  					 State#state.poll_limit), -    log(State,recv,"Return: ~p",[{ok,Data}]), +    log(State,recv,"Return: ~tp",[{ok,Data}]),      end_gen_log(),      {{ok,Data},State#state{buffer=Buffer}};  handle_msg({expect,Pattern,Opts},State) ->      start_gen_log(heading(expect,State#state.name)), -    log(State,expect,"Expect: ~p\nOpts = ~p\n",[Pattern,Opts]), +    log(State,expect,"Expect: ~tp\nOpts = ~tp\n",[Pattern,Opts]),      {Return,NewBuffer,Prompt} =   	case teln_expect(State#state.name,  			 State#state.teln_pid, @@ -779,15 +779,15 @@ handle_msg({expect,Pattern,Opts},State) ->  		P = check_if_prompt_was_reached(Data,[]),  		{{ok,Data},Rest,P};  	    {ok,Data,HaltReason,Rest} -> -		force_log(State,expect,"HaltReason: ~p",[HaltReason]), +		force_log(State,expect,"HaltReason: ~tp",[HaltReason]),  		P = check_if_prompt_was_reached(Data,HaltReason),  		{{ok,Data,HaltReason},Rest,P};  	    {error,Reason,Rest} -> -		force_log(State,expect,"Expect failed\n~p",[{error,Reason}]), +		force_log(State,expect,"Expect failed\n~tp",[{error,Reason}]),  		P = check_if_prompt_was_reached([],Reason),  		{{error,Reason},Rest,P};  	    {error,Reason} -> -		force_log(State,expect,"Expect failed\n~p",[{error,Reason}]), +		force_log(State,expect,"Expect failed\n~tp",[{error,Reason}]),  		P = check_if_prompt_was_reached([],Reason),  		{{error,Reason},[],P}  	end, @@ -896,7 +896,7 @@ check_if_prompt_was_reached(_,_) ->  heading(Action,undefined) ->      io_lib:format("~w ~w",[?MODULE,Action]);  heading(Action,Name) -> -    io_lib:format("~w ~w for ~p",[?MODULE,Action,Name]). +    io_lib:format("~w ~w for ~tp",[?MODULE,Action,Name]).  force_log(State,Action,String,Args) ->      log(State,Action,String,Args,true). @@ -1294,7 +1294,7 @@ get_data1(Pid) ->  %% one_expect: split data chunk at prompts  one_expect(Name,Pid,Data,Pattern,EO) when EO#eo.prompt_check==false -> -%    io:format("Raw Data ~p Pattern ~p EO ~p ",[Data,Pattern,EO]), +%    io:format("Raw Data ~tp Pattern ~tp EO ~tp ",[Data,Pattern,EO]),      one_expect1(Name,Pid,Data,Pattern,[],EO#eo{found_prompt=false});  one_expect(Name,Pid,Data,Pattern,EO) ->      case match_prompt(Data,EO#eo.prx) of @@ -1455,7 +1455,7 @@ match_line(Name,Pid,Line,[{prompt,PromptType}|Patterns],FoundPrompt,Term,    when PromptType=/=FoundPrompt ->      match_line(Name,Pid,Line,Patterns,FoundPrompt,Term,EO,RetTag);  match_line(Name,Pid,Line,[{Tag,Pattern}|Patterns],FoundPrompt,Term,EO,RetTag) -> -    case re:run(Line,Pattern,[{capture,all,list}]) of +    case re:run(Line,Pattern,[{capture,all,list},unicode]) of  	nomatch ->  	    match_line(Name,Pid,Line,Patterns,FoundPrompt,Term,EO,RetTag);  	{match,Match} -> @@ -1463,7 +1463,7 @@ match_line(Name,Pid,Line,[{Tag,Pattern}|Patterns],FoundPrompt,Term,EO,RetTag) ->  	    {RetTag,{Tag,Match}}      end;  match_line(Name,Pid,Line,[Pattern|Patterns],FoundPrompt,Term,EO,RetTag) -> -    case re:run(Line,Pattern,[{capture,all,list}]) of +    case re:run(Line,Pattern,[{capture,all,list},unicode]) of  	nomatch ->  	    match_line(Name,Pid,Line,Patterns,FoundPrompt,Term,EO,RetTag);  	{match,Match} -> @@ -1575,7 +1575,7 @@ split_lines([],Line,Lines) ->  match_prompt(Str,Prx) ->      match_prompt(Str,Prx,[]).  match_prompt(Str,Prx,Acc) -> -    case re:run(Str,Prx) of +    case re:run(Str,Prx,[unicode]) of  	nomatch ->  	    noprompt;  	{match,[{Start,Len}]} -> diff --git a/lib/common_test/src/ct_telnet_client.erl b/lib/common_test/src/ct_telnet_client.erl index 5df7e279ac..c8d217cd2a 100644 --- a/lib/common_test/src/ct_telnet_client.erl +++ b/lib/common_test/src/ct_telnet_client.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2003-2016. All Rights Reserved. +%% Copyright Ericsson AB 2003-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -120,7 +120,7 @@ get_data(Pid) ->  init(Parent, Server, Port, Timeout, KeepAlive, NoDelay, ConnName) ->      case gen_tcp:connect(Server, Port, [list,{packet,0},{nodelay,NoDelay}], Timeout) of  	{ok,Sock} -> -	    dbg("~p connected to: ~p (port: ~w, keep_alive: ~w)\n", +	    dbg("~tp connected to: ~tp (port: ~w, keep_alive: ~w)\n",  		[ConnName,Server,Port,KeepAlive]),  	    send([?IAC,?DO,?SUPPRESS_GO_AHEAD], Sock, ConnName),	        	    Parent ! {open,self()}, diff --git a/lib/common_test/src/ct_testspec.erl b/lib/common_test/src/ct_testspec.erl index 180786273d..bb445bb0d2 100644 --- a/lib/common_test/src/ct_testspec.erl +++ b/lib/common_test/src/ct_testspec.erl @@ -344,7 +344,7 @@ create_spec_tree([Spec|Specs],TS,JoinWithNext,Known) ->  		     create_spec_tree(Specs,TS,JoinWithNext,Known)};	  		{error,Reason} ->  		    ReasonStr = -			lists:flatten(io_lib:format("~s", +			lists:flatten(io_lib:format("~ts",  						    [file:format_error(Reason)])),  		    throw({error,{SpecAbsName,ReasonStr}})  	    end @@ -537,7 +537,7 @@ replace_names_in_elems([],Modified,_Defs) ->  replace_names_in_string(Term,Defs=[{Name,Replacement=[Ch|_]}|Ds])    when is_integer(Ch) ->      try re:replace(Term,[$'|atom_to_list(Name)]++"'", -		   Replacement,[{return,list}]) of +		   Replacement,[{return,list},unicode]) of  	Term ->					% no match, proceed  	    replace_names_in_string(Term,Ds);  	Term1 -> @@ -569,7 +569,7 @@ replace_names_in_node1(NodeStr,Defs=[{Name,Replacement}|Ds]) ->  	    replace_names_in_node1(NodeStr,Ds);         true ->  	    case re:replace(NodeStr,atom_to_list(Name), -			    ReplStr,[{return,list}]) of +			    ReplStr,[{return,list},unicode]) of  		NodeStr ->			% no match, proceed  		    replace_names_in_node1(NodeStr,Ds);  		NodeStr1 -> @@ -1101,7 +1101,7 @@ check_term(Term) when is_tuple(Term) ->  				true ->  				    io:format("~nSuspicious term, "  					      "please check:~n" -					      "~p~n", [Term]), +					      "~tp~n", [Term]),  				    invalid;  				false ->  				    invalid diff --git a/lib/common_test/src/ct_util.erl b/lib/common_test/src/ct_util.erl index 4d3a2ae7e3..abf131f4df 100644 --- a/lib/common_test/src/ct_util.erl +++ b/lib/common_test/src/ct_util.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2003-2016. All Rights Reserved. +%% Copyright Ericsson AB 2003-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -201,20 +201,13 @@ do_start(Parent, Mode, LogDir, Verbosity) ->  	ok ->  	    Parent ! {self(),started};  	{fail,CTHReason} -> -	    ErrorInfo = if is_atom(CTHReason) -> -				io_lib:format("{~p,~p}", -					      [CTHReason, -					       erlang:get_stacktrace()]); -			   true -> -				CTHReason -			end, -	    ct_logs:tc_print('Suite Callback',ErrorInfo,[]), +	    ct_logs:tc_print('Suite Callback',CTHReason,[]),  	    self() ! {{stop,{self(),{user_error,CTHReason}}},  		      {Parent,make_ref()}}      catch  	_:CTHReason ->  	    ErrorInfo = if is_atom(CTHReason) -> -				io_lib:format("{~p,~p}", +				io_lib:format("{~tp,~tp}",  					      [CTHReason,  					       erlang:get_stacktrace()]);  			   true -> @@ -497,7 +490,7 @@ loop(Mode,TestData,StartDir) ->  					 ?MAX_IMPORTANCE,  					 "CT Error Notification",  					 "Connection process died: " -					 "Pid: ~w, Address: ~p, " +					 "Pid: ~w, Address: ~tp, "  					 "Callback: ~w\n"  					 "Reason: ~ts\n\n",  					 [Pid,A,CB,ErrorHtml]), @@ -508,7 +501,7 @@ loop(Mode,TestData,StartDir) ->  		_ ->  		    %% Let process crash in case of error, this shouldn't happen!  		    io:format("\n\nct_util_server got EXIT " -			      "from ~w: ~p\n\n", [Pid,Reason]), +			      "from ~w: ~tp\n\n", [Pid,Reason]),  		    ok = file:set_cwd(StartDir),  		    exit(Reason)  	    end @@ -984,12 +977,12 @@ get_profile_data(Profile, Key, StartDir) ->  	end,      case Result of  	{error,enoent} when Profile /= default -> -	    io:format(?def_gl, "~nERROR! Missing profile file ~p~n", [File]), +	    io:format(?def_gl, "~nERROR! Missing profile file ~tp~n", [File]),  	    undefined;  	{error,enoent} when Profile == default ->  	    undefined;  	{error,Reason} -> -	    io:format(?def_gl,"~nERROR! Error in profile file ~p: ~p~n", +	    io:format(?def_gl,"~nERROR! Error in profile file ~tp: ~tp~n",  		      [WhichFile,Reason]),  	    undefined;  	{ok,Data} -> @@ -1000,7 +993,7 @@ get_profile_data(Profile, Key, StartDir) ->  			    Data;  			_ ->  			    io:format(?def_gl, -				      "~nERROR! Invalid profile data in ~p~n", +				      "~nERROR! Invalid profile data in ~tp~n",  				      [WhichFile]),  			    []  		    end, diff --git a/lib/common_test/src/ct_webtool.erl b/lib/common_test/src/ct_webtool.erl index 87af442fd3..9016aca899 100644 --- a/lib/common_test/src/ct_webtool.erl +++ b/lib/common_test/src/ct_webtool.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2001-2016. All Rights Reserved. +%% Copyright Ericsson AB 2001-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -120,10 +120,10 @@ debug_app(Mod) ->  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)]), +    io:format("~w: (~p)~ncall ~ts~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]), +    io:format("~w: (~p)~nreturned from ~ts -> ~tp~n", [TS,Pid,ffunc(MFA),R]),      S;  out(_,_,_,_) ->      ok. @@ -171,7 +171,7 @@ script_start([App,Browser]) ->  		    IExplore=filename:join(win32reg:expand(Val),"iexplore.exe"),                      os:cmd("\"" ++ IExplore ++ "\" " ++ Url);  		_ when OSType == win32 -> -                    io:format("Starting ~w...\n",[Browser]), +                    io:format("Starting ~tw...\n",[Browser]),                      os:cmd("\"" ++ atom_to_list(Browser) ++ "\" " ++ Url);  		B when B==firefox; B==mozilla ->  		    io:format("Sending URL to ~w...",[Browser]), @@ -194,7 +194,7 @@ script_start([App,Browser]) ->  			    os:cmd(BStr ++ " " ++ Url)  		    end;  		_ -> -		    io:format("Starting ~w...\n",[Browser]), +		    io:format("Starting ~tw...\n",[Browser]),  		    os:cmd(atom_to_list(Browser) ++ " " ++ Url)  	    end,  	    ok; @@ -379,7 +379,7 @@ 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("WebTool is available at http://~ts:~w/~n",[Server,Port]),      io:format("Or  http://~w.~w.~w.~w:~w/~n",[A,B,C,D,Port]). @@ -859,8 +859,8 @@ handle_app({Name,{start,{func,Start,Stop}}},Data,_Pid,Cmd)->  		    %%! 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", +			      "~w:~tw(~ts) ->\n" +			      "~tp\n\n",  			      [?LINE,Name,M,F,format_args(A),Exit]),  		    ets:delete(Data,Name);  		_OK-> @@ -883,16 +883,16 @@ handle_app({Name,{start,{child,ChildSpec}}},Data,Pid,Cmd)->  		    %%! 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", +			      "supervisor:start_child(~p,~tp) ->\n" +			      "~tp\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", +			      "supervisor:start_child(~p,~tp) ->\n" +			      "~tp\n\n",  			      [?LINE,Name,Pid,ChildSpec,Error]),  		    ets:delete(Data,Name)  	    end; @@ -924,7 +924,7 @@ handle_app({Name,{start,{app,Real_name}}},Data,_Pid,Cmd)->  		    io:format("\n=======ERROR (webtool, line ~w) =======\n"  			      "Could not start application \'~p\'\n\n"  			      "application:start(~p,~p) ->\n" -			      "~p\n\n", +			      "~tp\n\n",  			      [?LINE,Name,Real_name,temporary,Error]),  		    ets:delete(Data,Name)  	    end; @@ -940,7 +940,7 @@ 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", +	      "Incorrect data: ~tp\n\n",  	      [?LINE,Cmd,Name,Incorrect]),      ets:delete(Data,Name). @@ -1202,12 +1202,12 @@ filter_tool_files(Dir,[File|Rest]) ->  %%%-----------------------------------------------------------------  %%% format functions  ffunc({M,F,A}) when is_list(A) -> -    io_lib:format("~w:~w(~s)\n",[M,F,format_args(A)]); +    io_lib:format("~w:~tw(~ts)\n",[M,F,format_args(A)]);  ffunc({M,F,A}) when is_integer(A) -> -    io_lib:format("~w:~w/~w\n",[M,F,A]). +    io_lib:format("~w:~tw/~w\n",[M,F,A]).  format_args([]) ->      "";  format_args(Args) -> -    Str = lists:append(["~p"|lists:duplicate(length(Args)-1,",~p")]), +    Str = lists:append(["~tp"|lists:duplicate(length(Args)-1,",~tp")]),      io_lib:format(Str,Args). diff --git a/lib/common_test/src/cth_conn_log.erl b/lib/common_test/src/cth_conn_log.erl index ef92532969..7b6f03311a 100644 --- a/lib/common_test/src/cth_conn_log.erl +++ b/lib/common_test/src/cth_conn_log.erl @@ -24,11 +24,11 @@  %%  %% suite() ->  %%    [{ct_hooks, [{cth_conn_log, -%%                  [{ct_netconfc:conn_mod(),ct_netconfc:hook_options()}]}]}]. +%%                  [{conn_mod(),hook_options()}]}]}].  %%  %% or specified in a configuration file:  %% -%% {ct_conn_log,[{ct_netconfc:conn_mod(),ct_netconfc:hook_options()}]}. +%% {ct_conn_log,[{conn_mod(),hook_options()}]}.  %%  %% The conn_mod() is the common test module implementing the protocol,  %% e.g. ct_netconfc, ct_telnet, etc. This module must log by calling @@ -58,28 +58,17 @@  	 post_end_per_testcase/5]).  %%---------------------------------------------------------------------- -%% Exported types -%%---------------------------------------------------------------------- --export_type([hook_options/0, -	      log_type/0, -	      conn_mod/0]). - -%%----------------------------------------------------------------------  %% Type declarations  %%---------------------------------------------------------------------- --type hook_options() :: [hook_option()]. -%% Options that can be given to `cth_conn_log' in the `ct_hook' statement. --type hook_option() :: {log_type,log_type()} | -		       {hosts,[ct_gen_conn:key_or_name()]}. --type log_type() :: raw | pretty | html | silent. --type conn_mod() :: ct_netconfc | ct_telnet. +-type hook_options() :: ct:conn_log_options(). +-type log_type() :: ct:conn_log_type(). +-type conn_mod() :: ct:conn_log_mod().  %%----------------------------------------------------------------------  -spec init(Id, HookOpts) -> Result when        Id :: term(),        HookOpts :: hook_options(), -      Result :: {ok,[{conn_mod(), -		      {log_type(),[ct_gen_conn:key_or_name()]}}]}. +      Result :: {ok,[{conn_mod(),{log_type(),[ct:key_or_name()]}}]}.  init(_Id, HookOpts) ->      ConfOpts = ct:get_config(ct_conn_log,[]),      {ok,merge_log_info(ConfOpts,HookOpts)}. @@ -127,7 +116,7 @@ pre_init_per_testcase(_Suite,TestCase,Config,CthState) ->  			      "<table borders=1>"  			      "<b>" ++ ConnModStr ++ " logs:</b>\n" ++  			      [io_lib:format( -				 "<tr><td>~p</td><td><a href=\"~ts\">~ts</a>" +				 "<tr><td>~tp</td><td><a href=\"~ts\">~ts</a>"  				 "</td></tr>",  				 [S,ct_logs:uri(L),filename:basename(L)])  			       || {S,L} <- Ls] ++ diff --git a/lib/common_test/src/cth_log_redirect.erl b/lib/common_test/src/cth_log_redirect.erl index 1bc9b10d41..8b29d0f96d 100644 --- a/lib/common_test/src/cth_log_redirect.erl +++ b/lib/common_test/src/cth_log_redirect.erl @@ -250,26 +250,26 @@ format_header(#eh_state{curr_suite = Suite,  format_header(#eh_state{curr_suite = Suite,  			curr_group = undefined,  			curr_func = TcOrConf}) -> -    io_lib:format("System report during ~w:~w/1", +    io_lib:format("System report during ~w:~tw/1",  		  [Suite,TcOrConf]);  format_header(#eh_state{curr_suite = Suite,  			curr_group = Group,  			curr_func = Conf}) when Conf == init_per_group;  						Conf == end_per_group -> -    io_lib:format("System report during ~w:~w/2 for ~w", +    io_lib:format("System report during ~w:~w/2 for ~tw",  		  [Suite,Conf,Group]);  format_header(#eh_state{curr_suite = Suite,  			curr_group = Group,  			parallel_tcs = true}) -> -    io_lib:format("System report during ~w in ~w", +    io_lib:format("System report during ~tw in ~w",  		  [Group,Suite]);  format_header(#eh_state{curr_suite = Suite,  			curr_group = Group,  			curr_func = TC}) -> -    io_lib:format("System report during ~w:~w/1 in ~w", +    io_lib:format("System report during ~w:~tw/1 in ~tw",  		  [Suite,TC,Group]).  code_change(_OldVsn, State, _Extra) -> diff --git a/lib/common_test/src/cth_surefire.erl b/lib/common_test/src/cth_surefire.erl index 0bbb217275..da68bd105e 100644 --- a/lib/common_test/src/cth_surefire.erl +++ b/lib/common_test/src/cth_surefire.erl @@ -143,7 +143,7 @@ on_tc_fail(_Suite,_TC, Res, State) ->      TC = hd(TCs),      NewTC = TC#testcase{  	      result = -		  {fail,lists:flatten(io_lib:format("~p",[Res]))} }, +		  {fail,lists:flatten(io_lib:format("~tp",[Res]))} },      State#state{ test_cases = [NewTC | tl(TCs)]}.  on_tc_skip(Suite,{ConfigFunc,_GrName}, Res, State) -> @@ -164,7 +164,7 @@ do_tc_skip(Res, State) ->      TC = hd(TCs),      NewTC = TC#testcase{  	      result = -		  {skipped,lists:flatten(io_lib:format("~p",[Res]))} }, +		  {skipped,lists:flatten(io_lib:format("~tp",[Res]))} },      State#state{ test_cases = [NewTC | tl(TCs)]}.  init_tc(State, Config) when is_list(Config) == false -> diff --git a/lib/common_test/src/test_server.erl b/lib/common_test/src/test_server.erl index 857ff27258..dc6b7a536c 100644 --- a/lib/common_test/src/test_server.erl +++ b/lib/common_test/src/test_server.erl @@ -21,7 +21,7 @@  -define(DEFAULT_TIMETRAP_SECS, 60).  %%% TEST_SERVER_CTRL INTERFACE %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% --export([run_test_case_apply/1,init_target_info/0]). +-export([run_test_case_apply/1,init_target_info/0,init_valgrind/0]).  -export([cover_compile/1,cover_analyse/2]).  %%% TEST_SERVER_SUP INTERFACE %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% @@ -49,6 +49,10 @@  -export([break/1,break/2,break/3,continue/0,continue/1]). +%%% DEBUGGER INTERFACE %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +-export([valgrind_new_leaks/0, valgrind_format/2, +	 is_valgrind/0]). +  %%% PRIVATE EXPORTED %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  -export([]). @@ -69,6 +73,10 @@ init_target_info() ->  		 username=test_server_sup:get_username(),  		 cookie=atom_to_list(erlang:get_cookie())}. +init_valgrind() -> +    valgrind_new_leaks(). + +  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %% cover_compile(#cover{app=App,incl=Include,excl=Exclude,cross=Cross}) ->  %%        {ok,#cover{mods=AnalyseModules}} | {error,Reason} @@ -175,7 +183,7 @@ do_cover_compile(Modules) ->      ok.  warn_compile({error,{Reason,Module}}) -> -    io:fwrite("\nWARNING: Could not cover compile ~ts: ~p\n", +    io:fwrite("\nWARNING: Could not cover compile ~ts: ~tp\n",  	      [Module,{error,Reason}]).  %% Make sure all modules are loaded and unstick if sticky @@ -189,7 +197,7 @@ prepare_cover_compile([M|Ms],Sticky) ->  		{module,_} ->  		    prepare_cover_compile([M|Ms],Sticky);  		Error -> -		    io:fwrite("\nWARNING: Could not load ~w: ~p\n",[M,Error]), +		    io:fwrite("\nWARNING: Could not load ~w: ~tp\n",[M,Error]),  		    prepare_cover_compile(Ms,Sticky)  	    end;  	{false,_} -> @@ -358,11 +366,12 @@ stick_all_sticky(Node,Sticky) ->  %% compensate timetraps for runtime delays introduced by e.g. tools like  %% cover. -run_test_case_apply({Mod,Func,Args,Name,RunInit,TimetrapData}) -> -    case os:getenv("TS_RUN_VALGRIND") of +run_test_case_apply({CaseNum,Mod,Func,Args,Name,RunInit,TimetrapData}) -> +    case is_valgrind() of  	false ->  	    ok; -	_ -> +	true -> +            valgrind_format("Test case #~w ~w:~w/1", [CaseNum, Mod, Func]),  	    os:putenv("VALGRIND_LOGFILE_INFIX",atom_to_list(Mod)++"."++  		      atom_to_list(Func)++"-")      end, @@ -370,6 +379,7 @@ run_test_case_apply({Mod,Func,Args,Name,RunInit,TimetrapData}) ->      Result = run_test_case_apply(Mod, Func, Args, Name, RunInit,  				 TimetrapData),      ProcAft = erlang:system_info(process_count), +    valgrind_new_leaks(),      DetFail = get(test_server_detected_fail),      {Result,DetFail,ProcBef,ProcAft}. @@ -450,7 +460,7 @@ run_test_case_msgloop(#st{ref=Ref,pid=Pid,end_conf_pid=EndConfPid0}=St0) ->  			 exit(Pid, kill),  			 %% here's the only place we know Reason, so we save  			 %% it as a comment, potentially replacing user data -			 Error = lists:flatten(io_lib:format("Aborted: ~p", +			 Error = lists:flatten(io_lib:format("Aborted: ~tp",  							     [Reason])),  			 Error1 = lists:flatten([string:strip(S,left) ||  						    S <- string:tokens(Error, @@ -756,13 +766,13 @@ print_end_conf_result(Mod,Func,Conf,Cause,Error) ->      Str2Print =  	fun(NoHTML) when NoHTML == stdout; NoHTML == major ->  		io_lib:format("WARNING! " -			      "~w:end_per_testcase(~w, ~tp)" +			      "~w:end_per_testcase(~tw, ~tp)"  			      " ~s!\n\tReason: ~tp\n",  			      [Mod,Func,Conf,Cause,Error]);		      	   (minor) ->  		ErrorStr = test_server_ctrl:escape_chars(Error),  		io_lib:format("WARNING! " -			      "~w:end_per_testcase(~w, ~tp)" +			      "~w:end_per_testcase(~tw, ~tp)"  			      " ~s!\n\tReason: ~ts\n",  			      [Mod,Func,Conf,Cause,ErrorStr])  	end, @@ -792,7 +802,7 @@ spawn_fw_call(Mod,IPTC={init_per_testcase,Func},CurrConf,Pid,  			   _                       -> died  		       end,  		group_leader() ! {printout,12, -				  "ERROR! ~w:init_per_testcase(~w, ~p)" +				  "ERROR! ~w:init_per_testcase(~tw, ~tp)"  				  " failed!\n\tReason: ~tp\n",  				 [Mod,Func,CurrConf,Why]},  		%% finished, report back @@ -820,7 +830,7 @@ spawn_fw_call(Mod,EPTC={end_per_testcase,Func},EndConf,Pid,  			{timetrap_timeout,TVal} ->  			    group_leader() !  				{printout,12, -				 "WARNING! ~w:end_per_testcase(~w, ~p)" +				 "WARNING! ~w:end_per_testcase(~tw, ~tp)"  				 " failed!\n\tReason: timetrap timeout"  				 " after ~w ms!\n", [Mod,Func,EndConf,TVal]},  			    W = "<font color=\"red\">" @@ -829,7 +839,7 @@ spawn_fw_call(Mod,EPTC={end_per_testcase,Func},EndConf,Pid,  			_ ->  			    group_leader() !  				{printout,12, -				 "WARNING! ~w:end_per_testcase(~w, ~p)" +				 "WARNING! ~w:end_per_testcase(~tw, ~tp)"  				 " failed!\n\tReason: ~tp\n",  				 [Mod,Func,EndConf,Why]},  			    W = "<font color=\"red\">" @@ -859,7 +869,7 @@ spawn_fw_call(FwMod,FwFunc,_,_Pid,{framework_error,FwError},_,SendTo) ->  		Comment =  		    lists:flatten(  		      io_lib:format("<font color=\"red\">" -				    "WARNING! ~w:~w failed!</font>", +				    "WARNING! ~w:~tw failed!</font>",  				    [FwMod,FwFunc])),  	    %% finished, report back  	    SendTo ! {self(),fw_notify_done, @@ -1341,7 +1351,7 @@ print_init_conf_result(Line,Cause,Reason) ->      Str2Print =  	fun(NoHTML) when NoHTML == stdout; NoHTML == major ->  		io_lib:format("ERROR! init_per_testcase ~s!\n" -				      "\tLocation: ~p\n\tReason: ~tp\n", +				      "\tLocation: ~tp\n\tReason: ~tp\n",  				      [Cause,Line,Reason]);  	   (minor) ->  		ReasonStr = test_server_ctrl:escape_chars(Reason), @@ -1413,7 +1423,7 @@ print_end_tc_warning(EndFunc,Reason,Cause,Loc) ->      Str2Print =  	fun(NoHTML) when NoHTML == stdout; NoHTML == major ->  		io_lib:format("WARNING: ~w ~s!\n" -			      "Reason: ~tp\nLine: ~p\n", +			      "Reason: ~tp\nLine: ~tp\n",  			      [EndFunc,Cause,Reason,Loc]);  	   (minor) ->  		ReasonStr = test_server_ctrl:escape_chars(Reason), @@ -1515,7 +1525,7 @@ lookup_config(Key,Config) ->  	{value,{Key,Val}} ->  	    Val;  	_ -> -	    io:format("Could not find element ~p in Config.~n",[Key]), +	    io:format("Could not find element ~tp in Config.~n",[Key]),  	    undefined      end. @@ -1600,7 +1610,7 @@ format(Detail, Format, Args) ->      Str =  	case catch io_lib:format(Format,Args) of  	    {'EXIT',_} -> -		io_lib:format("illegal format; ~p with args ~p.\n", +		io_lib:format("illegal format; ~tp with args ~tp.\n",  			      [Format,Args]);  	    Valid -> Valid  	end, @@ -1732,7 +1742,7 @@ fail(Reason) ->  cast_to_list(X) when is_list(X) -> X;  cast_to_list(X) when is_atom(X) -> atom_to_list(X); -cast_to_list(X) -> lists:flatten(io_lib:format("~p", [X])). +cast_to_list(X) -> lists:flatten(io_lib:format("~tp", [X])). @@ -1827,7 +1837,8 @@ timetrap_scale_factor() ->  	{ 2, fun() -> has_lock_checking() end},  	{ 3, fun() -> has_superfluous_schedulers() end},  	{ 6, fun() -> is_debug() end}, -	{10, fun() -> is_cover() end} +	{10, fun() -> is_cover() end}, +        {10, fun() -> is_valgrind() end}      ]).  timetrap_scale_factor(Scales) -> @@ -1903,7 +1914,7 @@ ensure_timetrap(Config) ->  		Garbage ->  		    erase(test_server_default_timetrap),  		    format("=== WARNING: garbage in " -			   "test_server_default_timetrap: ~p~n", +			   "test_server_default_timetrap: ~tp~n",  			   [Garbage])  	    end,  	    DTmo = case lists:keysearch(default_timeout,1,Config) of @@ -1932,7 +1943,7 @@ cancel_default_timetrap(true) ->  	Garbage ->  	    erase(test_server_default_timetrap),  	    format("=== WARNING: garbage in " -		   "test_server_default_timetrap: ~p~n", +		   "test_server_default_timetrap: ~tp~n",  		   [Garbage]),  	    error      end. @@ -1941,7 +1952,7 @@ time_ms({hours,N}, _, _) -> hours(N);  time_ms({minutes,N}, _, _) -> minutes(N);  time_ms({seconds,N}, _, _) -> seconds(N);  time_ms({Other,_N}, _, _) -> -    format("=== ERROR: Invalid time specification: ~p. " +    format("=== ERROR: Invalid time specification: ~tp. "  	   "Should be seconds, minutes, or hours.~n", [Other]),      exit({invalid_time_format,Other});  time_ms(Ms, _, _) when is_integer(Ms) -> Ms; @@ -2729,6 +2740,41 @@ is_commercial() ->  	_ -> true      end. +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% is_valgrind() -> boolean() +%% +%% Returns true if valgrind is running, else false +is_valgrind() -> +    case catch erlang:system_info({valgrind, running}) of +	{'EXIT', _} -> false; +	Res -> Res +    end. + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%%                     DEBUGGER INTERFACE                    %% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% valgrind_new_leaks() -> ok +%% +%% Checks for new memory leaks if Valgrind is active. +valgrind_new_leaks() -> +    catch erlang:system_info({valgrind, memory}), +    ok. + +%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% +%% valgrind_format(Format, Args) -> ok +%% Format = string() +%% Args = lists() +%% +%% Outputs the formatted string to Valgrind's logfile,if Valgrind is active. +valgrind_format(Format, Args) -> +    (catch erlang:system_info({valgrind, io_lib:format(Format, Args)})), +    ok. + + +  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %%  %% Apply given function and reply to caller or proxy. diff --git a/lib/common_test/src/test_server_ctrl.erl b/lib/common_test/src/test_server_ctrl.erl index 064e375cd5..71978c7267 100644 --- a/lib/common_test/src/test_server_ctrl.erl +++ b/lib/common_test/src/test_server_ctrl.erl @@ -232,7 +232,7 @@ parse_cmd_line(['SPEC',Spec|Cmds], SpecList, Names, Param, Trc, Cov, TCCB) ->  	    parse_cmd_line(Cmds, TermList++SpecList, [Name|Names], Param,  			   Trc, Cov, TCCB);  	{error,Reason} -> -	    io:format("Can't open ~w: ~p\n",[Spec, file:format_error(Reason)]), +	    io:format("Can't open ~tw: ~tp\n",[Spec, file:format_error(Reason)]),  	    parse_cmd_line(Cmds, SpecList, Names, Param, Trc, Cov, TCCB)      end;  parse_cmd_line(['NAME',Name|Cmds], SpecList, Names, Param, Trc, Cov, TCCB) -> @@ -261,7 +261,7 @@ parse_cmd_line(['COVER',App,CF,Analyse|Cmds], SpecList, Names, Param, Trc, _Cov,  parse_cmd_line(['TESTCASE_CALLBACK',Mod,Func|Cmds], SpecList, Names, Param, Trc, Cov, _) ->      parse_cmd_line(Cmds, SpecList, Names, Param, Trc, Cov, {Mod,Func});  parse_cmd_line([Obj|_Cmds], _SpecList, _Names, _Param, _Trc, _Cov, _TCCB) -> -    io:format("~w: Bad argument: ~w\n", [?MODULE,Obj]), +    io:format("~w: Bad argument: ~tw\n", [?MODULE,Obj]),      io:format(" Use the `ts' module to start tests.\n", []),      io:format(" (If you ARE using `ts', there is a bug in `ts'.)\n", []),      halt(1); @@ -280,7 +280,7 @@ parse_cmd_line([], SpecList, Names, Param, Trc, Cov, TCCB) ->  cast_to_list(X) when is_list(X) -> X;  cast_to_list(X) when is_atom(X) -> atom_to_list(X); -cast_to_list(X) -> lists:flatten(io_lib:format("~w", [X])). +cast_to_list(X) -> lists:flatten(io_lib:format("~tw", [X])).  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %% START INTERFACE @@ -878,7 +878,7 @@ handle_call({testcase_callback,ModFunc}, _From, State) ->  		    ok;  		false ->  		    io:format(user, -			      "WARNING! Callback function ~w:~w/4 undefined.~n~n", +			      "WARNING! Callback function ~w:~tw/4 undefined.~n~n",  			      [Mod,Func])  	    end;  	_ -> @@ -1016,7 +1016,7 @@ handle_info({'EXIT',Pid,Reason}, State) ->  		killed ->  		    io:format("Suite ~ts was killed\n", [Name]);  		_Other -> -		    io:format("Suite ~ts was killed with reason ~p\n", +		    io:format("Suite ~ts was killed with reason ~tp\n",  			      [Name,Reason])  	    end,  	    State2 = State#state{jobs=NewJobs}, @@ -1168,10 +1168,10 @@ init_tester(Mod, Func, Args, Dir, Name, {_,_,MinLev}=Levels,  	{'EXIT',test_suites_done} ->  	    ok;  	{'EXIT',_Pid,Reason} -> -	    print(1, "EXIT, reason ~p", [Reason]); +	    print(1, "EXIT, reason ~tp", [Reason]);  	{'EXIT',Reason} ->  	    report_severe_error(Reason), -	    print(1, "EXIT, reason ~p", [Reason]) +	    print(1, "EXIT, reason ~tp", [Reason])      end,      Time = TimeMy/1000000,      SuccessStr = @@ -1263,7 +1263,7 @@ do_spec(SpecName, TimetrapSpec) when is_list(SpecName) ->  	{ok,TermList} ->  	    do_spec_list(TermList,TimetrapSpec);  	{error,Reason} -> -	    io:format("Can't open ~ts: ~p\n", [SpecName,Reason]), +	    io:format("Can't open ~ts: ~tp\n", [SpecName,Reason]),  	    {error,{cant_open_spec,Reason}}      end. @@ -1346,7 +1346,7 @@ do_spec_terms([{require_nodenames,NumNames}|Terms], TopCases, SkipList, Config)      do_spec_terms(Terms, TopCases, SkipList,  		  update_config(Config, {nodenames,NodeNames}));  do_spec_terms([Other|Terms], TopCases, SkipList, Config) -> -    io:format("** WARNING: Spec file contains unknown directive ~p\n", +    io:format("** WARNING: Spec file contains unknown directive ~tp\n",  	      [Other]),      do_spec_terms(Terms, TopCases, SkipList, Config). @@ -1503,7 +1503,7 @@ do_test_cases(TopCases, SkipCases,      FwMod = get_fw_mod(?MODULE),      case collect_all_cases(TopCases, SkipCases) of  	{error,Why} -> -	    print(1, "Error starting: ~p", [Why]), +	    print(1, "Error starting: ~tp", [Why]),  	    exit(test_suites_done);  	TestSpec0 ->  	    N = case remove_conf(TestSpec0) of @@ -1762,18 +1762,14 @@ make_html_link(LinkName, Target, Explanation) ->  start_minor_log_file(Mod, Func, ParallelTC) ->      MFA = {Mod,Func,1},      LogDir = get(test_server_log_dir_base), -    Name0 = lists:flatten(io_lib:format("~w.~w~ts", [Mod,Func,?html_ext])), -    Name = downcase(Name0), +    Name = minor_log_file_name(Mod,Func),      AbsName = filename:join(LogDir, Name),      case (ParallelTC orelse (element(1,file:read_file_info(AbsName))==ok)) of  	false ->                           %% normal case, unique name  	    start_minor_log_file1(Mod, Func, LogDir, AbsName, MFA);  	true ->                            %% special case, duplicate names  	    Tag = test_server_sup:unique_name(), -	    Name1_0 = -		lists:flatten(io_lib:format("~w.~w.~ts~ts", [Mod,Func,Tag, -							      ?html_ext])), -	    Name1 = downcase(Name1_0), +            Name1 = minor_log_file_name(Mod,Func,[$.|Tag]),  	    AbsName1 = filename:join(LogDir, Name1),  	    start_minor_log_file1(Mod, Func, LogDir, AbsName1, MFA)      end. @@ -1784,7 +1780,7 @@ start_minor_log_file1(Mod, Func, LogDir, AbsName, MFA) ->      put(test_server_minor_fd, Fd),      test_server_gl:set_minor_fd(group_leader(), Fd, MFA), -    TestDescr = io_lib:format("Test ~w:~w result", [Mod,Func]), +    TestDescr = io_lib:format("Test ~w:~tw result", [Mod,Func]),      {Header,Footer} =  	case test_server_sup:framework_call(get_html_wrapper,  					    [TestDescr,false, @@ -1825,13 +1821,13 @@ start_minor_log_file1(Mod, Func, LogDir, AbsName, MFA) ->  		  lists:member(no_src, get(test_server_logopts))} of  		{true,false} ->  		    print(Lev, ["$tc_html", -				Info ++ "<a href=\"~ts#~ts\">~w:~w/~w</a> " +				Info ++ "<a href=\"~ts#~ts\">~w:~tw/~w</a> "  				"(click for source code)\n"],  			  [uri_encode(SrcListing),  			   uri_encode(atom_to_list(Func)++"-1",utf8),  			   Mod,Func,Arity]);  		_ -> -		    print(Lev, ["$tc_html",Info ++ "~w:~w/~w\n"], [Mod,Func,Arity]) +		    print(Lev, ["$tc_html",Info ++ "~w:~tw/~w\n"], [Mod,Func,Arity])  	    end      end, @@ -1845,6 +1841,19 @@ stop_minor_log_file() ->      ok = file:close(Fd),      put(test_server_minor_fd, undefined). +minor_log_file_name(Mod,Func) -> +    minor_log_file_name(Mod,Func,""). +minor_log_file_name(Mod,Func,Tag) -> +    Name = +        downcase( +          lists:flatten( +            io_lib:format("~w.~tw~s~s", [Mod,Func,Tag,?html_ext]))), +    Ok = file:native_name_encoding()==utf8 +        orelse io_lib:printable_latin1_list(Name), +    if Ok -> Name; +       true -> exit({error,unicode_name_on_latin1_file_system}) +    end. +  downcase(S) -> downcase(S, []).  downcase([Uc|Rest], Result) when $A =< Uc, Uc =< $Z ->      downcase(Rest, [Uc-$A+$a|Result]); @@ -2154,6 +2163,7 @@ do_add_end_per_suite_and_skip(LastMod, LastRef, Mod, FwMod) ->  %% Runs the specified tests, then displays/logs the summary.  run_test_cases(TestSpec, Config, TimetrapData) -> +    test_server:init_valgrind(),      case lists:member(no_src, get(test_server_logopts)) of  	true ->  	    ok; @@ -2736,7 +2746,7 @@ run_test_cases_loop([{conf,Ref,Props,{Mod,Func}}|_Cases]=Cs0,  					TimetrapData, Mode, Status2);  		Bad ->  		    print(minor, -			  "~n*** ~w returned bad elements in Config: ~p.~n", +			  "~n*** ~tw returned bad elements in Config: ~tp.~n",  			  [Func,Bad]),  		    Reason = {failed,{Mod,init_per_suite,bad_return}},  		    Cases2 = skip_cases_upto(Ref, Cases, Reason, conf, CurrMode, @@ -2752,9 +2762,9 @@ run_test_cases_loop([{conf,Ref,Props,{Mod,Func}}|_Cases]=Cs0,  	    stop_minor_log_file(),  	    run_test_cases_loop(Cases, [NewCfg|Config], TimetrapData, Mode, Status2);  	{_,{framework_error,{FwMod,FwFunc},Reason},_} -> -	    print(minor, "~n*** ~w failed in ~w. Reason: ~p~n", +	    print(minor, "~n*** ~w failed in ~tw. Reason: ~tp~n",  		  [FwMod,FwFunc,Reason]), -	    print(1, "~w failed in ~w. Reason: ~p~n", [FwMod,FwFunc,Reason]), +	    print(1, "~w failed in ~tw. Reason: ~tp~n", [FwMod,FwFunc,Reason]),  	    exit(framework_error);  	{_,Fail,_} when element(1,Fail) == 'EXIT';  			element(1,Fail) == timetrap_timeout; @@ -2763,7 +2773,7 @@ run_test_cases_loop([{conf,Ref,Props,{Mod,Func}}|_Cases]=Cs0,  	    {Cases2,Config1,Status3} =  		if StartConf ->  			ReportAbortRepeat(failed), -			print(minor, "~n*** ~w failed.~n" +			print(minor, "~n*** ~tw failed.~n"  			      "    Skipping all cases.", [Func]),  			Reason = {failed,{Mod,Func,Fail}},  			{skip_cases_upto(Ref, Cases, Reason, conf, CurrMode, @@ -2786,7 +2796,7 @@ run_test_cases_loop([{conf,Ref,Props,{Mod,Func}}|_Cases]=Cs0,  	    {Cases2,Config1,Status3} =  		if StartConf ->  			ReportAbortRepeat(auto_skipped), -			print(minor, "~n*** ~w auto skipped.~n" +			print(minor, "~n*** ~tw auto skipped.~n"  			      "    Skipping all cases.", [Func]),  			{skip_cases_upto(Ref, Cases, SkipReason, conf, CurrMode,  					 auto_skip_case), @@ -2803,7 +2813,7 @@ run_test_cases_loop([{conf,Ref,Props,{Mod,Func}}|_Cases]=Cs0,  	{_,{Skip,Reason},_} when StartConf and ((Skip==skip) or (Skip==skipped)) ->  	    ReportAbortRepeat(skipped), -	    print(minor, "~n*** ~w skipped.~n" +	    print(minor, "~n*** ~tw skipped.~n"  		  "    Skipping all cases.", [Func]),  	    set_io_buffering(IOHandler),  	    stop_minor_log_file(), @@ -2813,7 +2823,7 @@ run_test_cases_loop([{conf,Ref,Props,{Mod,Func}}|_Cases]=Cs0,  				delete_status(Ref, Status2));  	{_,{skip_and_save,Reason,_SavedConfig},_} when StartConf ->  	    ReportAbortRepeat(skipped), -	    print(minor, "~n*** ~w skipped.~n" +	    print(minor, "~n*** ~tw skipped.~n"  		  "    Skipping all cases.", [Func]),  	    set_io_buffering(IOHandler),  	    stop_minor_log_file(), @@ -2878,7 +2888,7 @@ run_test_cases_loop([{make,Ref,{Mod,Func,Args}}|Cases0], Config, TimetrapData,  		    Mode, Status) ->      case run_test_case(Ref, 0, Mod, Func, Args, skip_init, TimetrapData) of  	{_,Why={'EXIT',_},_} -> -	    print(minor, "~n*** ~w failed.~n" +	    print(minor, "~n*** ~tw failed.~n"   		  "    Skipping all cases.", [Func]),  	    Reason = {failed,{Mod,Func,Why}},  	    Cases = skip_cases_upto(Ref, Cases0, Reason, conf, Mode, @@ -2932,9 +2942,9 @@ run_test_cases_loop([{Mod,Func,Args}|Cases], Config, TimetrapData, Mode, Status)  		       RunInit, TimetrapData, Mode) of  	%% callback to framework module failed, exit immediately  	{_,{framework_error,{FwMod,FwFunc},Reason},_} -> -	    print(minor, "~n*** ~w failed in ~w. Reason: ~p~n", +	    print(minor, "~n*** ~w failed in ~tw. Reason: ~tp~n",  		  [FwMod,FwFunc,Reason]), -	    print(1, "~w failed in ~w. Reason: ~p~n", [FwMod,FwFunc,Reason]), +	    print(1, "~w failed in ~tw. Reason: ~tp~n", [FwMod,FwFunc,Reason]),  	    stop_minor_log_file(),  	    exit(framework_error);  	%% sequential execution of test case finished @@ -2965,7 +2975,7 @@ run_test_cases_loop([{Mod,Func,Args}|Cases], Config, TimetrapData, Mode, Status)  			    stop_minor_log_file(),  			    run_test_cases_loop(Cases, Config, TimetrapData, Mode, Status1);  		       true ->	              % skip rest of cases in sequence -			    print(minor, "~n*** ~w failed.~n" +			    print(minor, "~n*** ~tw failed.~n"  				  "    Skipping all other cases in sequence.",  				  [Func]),  			    Reason = {failed,{Mod,Func}}, @@ -3105,8 +3115,8 @@ print_conf_time(ConfTime) ->  print_props([]) ->      ok;  print_props(Props) -> -    print(major, "=group_props   ~p", [Props]), -    print(minor, "Group properties: ~p~n", [Props]). +    print(major, "=group_props   ~tp", [Props]), +    print(minor, "Group properties: ~tp~n", [Props]).  %% repeat N times:                                  {repeat,N}  %% repeat N times or until all successful:          {repeat_until_all_ok,N} @@ -3253,13 +3263,13 @@ skip_case1(Type, CaseNum, Mod, Func, Comment, Mode) ->      ResultCol = if Type == auto -> ?auto_skip_color;  		   Type == user -> ?user_skip_color  		end, -    print(major, "~n=case          ~w:~w", [Mod,Func]), +    print(major, "~n=case          ~w:~tw", [Mod,Func]),      GroupName =	case get_name(Mode) of  		    undefined ->  			"";  		    GrName ->  			GrName1 = cast_to_list(GrName), -			print(major, "=group_props   ~p", [[{name,GrName1}]]), +			print(major, "=group_props   ~tp", [[{name,GrName1}]]),  			GrName1  		end,      print(major, "=started       ~s", [lists:flatten(timestamp_get(""))]), @@ -3270,9 +3280,9 @@ skip_case1(Type, CaseNum, Mod, Func, Comment, Mode) ->  	    print(major, "=result        skipped: ~ts", [Comment1])      end,      if CaseNum == 0 -> -	    print(2,"*** Skipping ~w ***", [{Mod,Func}]); +	    print(2,"*** Skipping ~tw ***", [{Mod,Func}]);         true -> -	    print(2,"*** Skipping test case #~w ~w ***", [CaseNum,{Mod,Func}]) +	    print(2,"*** Skipping test case #~w ~tw ***", [CaseNum,{Mod,Func}])      end,      TR = xhtml("<tr valign=\"top\">", ["<tr class=\"",odd_or_even(),"\">"]),	             GroupName =	case get_name(Mode) of @@ -3283,7 +3293,7 @@ skip_case1(Type, CaseNum, Mod, Func, Comment, Mode) ->  	  TR ++ "<td>" ++ Col0 ++ "~ts" ++ Col1 ++ "</td>"  	  "<td>" ++ Col0 ++ "~w" ++ Col1 ++ "</td>"  	  "<td>" ++ Col0 ++ "~ts" ++ Col1 ++ "</td>" -	  "<td>" ++ Col0 ++ "~w" ++ Col1 ++ "</td>" +	  "<td>" ++ Col0 ++ "~tw" ++ Col1 ++ "</td>"  	  "<td>" ++ Col0 ++ "< >" ++ Col1 ++ "</td>"  	  "<td>" ++ Col0 ++ "0.000s" ++ Col1 ++ "</td>"  	  "<td><font color=\"~ts\">SKIPPED</font></td>" @@ -3504,7 +3514,7 @@ wait_and_resend(Ref, [{_,CurrPid,CaseNum,Mod,Func}|Ps] = Cases, Ok,Skip,Fail) ->  	{'EXIT',CurrPid,Reason} when Reason /= normal ->  	    %% unexpected termination of test case process  	    {value,{_,_,CaseNum,Mod,Func}} = lists:keysearch(CurrPid, 2, Cases), -	    print(1, "Error! Process for test case #~w (~w:~w) died! Reason: ~p", +	    print(1, "Error! Process for test case #~w (~w:~tw) died! Reason: ~tp",  		  [CaseNum, Mod, Func, Reason]),  	    exit({unexpected_termination,{CaseNum,Mod,Func},{CurrPid,Reason}})      end; @@ -3643,7 +3653,7 @@ handle_io_and_exits(Main, CurrPid, CaseNum, Mod, Func, Cases) ->  	{'EXIT',TCPid,Reason} when Reason /= normal ->  	    test_server_io:print_buffered(CurrPid),  	    {value,{_,_,Num,M,F}} = lists:keysearch(TCPid, 2, Cases), -	    print(1, "Error! Process for test case #~w (~w:~w) died! Reason: ~p", +	    print(1, "Error! Process for test case #~w (~w:~tw) died! Reason: ~tp",  		  [Num, M, F, Reason]),  	    exit({unexpected_termination,{Num,M,F},{TCPid,Reason}})      end. @@ -3716,7 +3726,7 @@ run_test_case1(Ref, Num, Mod, Func, Args, RunInit,      end,      TSDir = get(test_server_dir), -    print(major, "=case          ~w:~w", [Mod, Func]), +    print(major, "=case          ~w:~tw", [Mod, Func]),      MinorName = start_minor_log_file(Mod, Func, self() /= Main),      MinorBase = filename:basename(MinorName),      print(major, "=logfile       ~ts", [filename:basename(MinorName)]), @@ -3778,7 +3788,7 @@ run_test_case1(Ref, Num, Mod, Func, Args, RunInit,      print(html,	TR ++ "<td>" ++ Col0 ++ "~ts" ++ Col1 ++ "</td>"  	  "<td>" ++ Col0 ++ "~w" ++ Col1 ++ "</td>"  	  "<td>" ++ Col0 ++ "~ts" ++ Col1 ++ "</td>" -	  "<td><a href=\"~ts\">~w</a></td>" +	  "<td><a href=\"~ts\">~tw</a></td>"  	  "<td><a href=\"~ts#top\"><</a> <a href=\"~ts#end\">></a></td>",  	  [num2str(Num),fw_name(Mod),GrNameStr,EncMinorBase,Func,  	   EncMinorBase,EncMinorBase]), @@ -3787,7 +3797,7 @@ run_test_case1(Ref, Num, Mod, Func, Args, RunInit,      %% run the test case      {Result,DetectedFail,ProcsBefore,ProcsAfter} = -	run_test_case_apply(Mod, Func, [UpdatedArgs], GrName, +	run_test_case_apply(Num, Mod, Func, [UpdatedArgs], GrName,  			    RunInit, TimetrapData),      {Time,RetVal,Loc,Opts,Comment} =  	case Result of @@ -3894,13 +3904,13 @@ run_test_case1(Ref, Num, Mod, Func, Args, RunInit,  			{'EXIT',_} = Exit ->  			    print(minor,  				  "WARNING: There might be slavenodes left in the" -				  " system. I tried to kill them, but I failed: ~p\n", +				  " system. I tried to kill them, but I failed: ~tp\n",  				  [Exit]);  			[] -> ok;  			List ->  			    print(minor, "WARNING: ~w slave nodes in system after test"++  				  "case. Tried to killed them.~n"++ -				  "         Names:~p", +				  "         Names:~tp",  				  [length(List),List])  		    end;  		false -> @@ -3960,7 +3970,7 @@ progress(skip, CaseNum, Mod, Func, GrName, Loc, Reason, Time,  	if_auto_skip(Reason,  		     fun() -> {?auto_skip_color,auto_skip,auto_skipped} end,  		     fun() -> {?user_skip_color,skip,skipped} end), -    print(major, "=result        ~w: ~p", [ReportTag,Reason1]), +    print(major, "=result        ~w: ~tp", [ReportTag,Reason1]),      print(1, "*** SKIPPED ~ts ***",  	  [get_info_str(Mod,Func, CaseNum, get(test_server_cases))]),      test_server_sup:framework_call(report, [tc_done,{Mod,{Func,GrName}, @@ -3993,7 +4003,7 @@ progress(skip, CaseNum, Mod, Func, GrName, Loc, Reason, Time,  progress(failed, CaseNum, Mod, Func, GrName, Loc, timetrap_timeout, T,  	 Comment0, {St0,St1}) -> -    print(major, "=result        failed: timeout, ~p", [Loc]), +    print(major, "=result        failed: timeout, ~tp", [Loc]),      print(1, "*** FAILED ~ts ***",  	  [get_info_str(Mod,Func, CaseNum, get(test_server_cases))]),      test_server_sup:framework_call(report, @@ -4019,7 +4029,7 @@ progress(failed, CaseNum, Mod, Func, GrName, Loc, timetrap_timeout, T,  progress(failed, CaseNum, Mod, Func, GrName, Loc, {testcase_aborted,Reason}, _T,  	 Comment0, {St0,St1}) -> -    print(major, "=result        failed: testcase_aborted, ~p", [Loc]), +    print(major, "=result        failed: testcase_aborted, ~tp", [Loc]),      print(1, "*** FAILED ~ts ***",  	  [get_info_str(Mod,Func, CaseNum, get(test_server_cases))]),      test_server_sup:framework_call(report, @@ -4041,14 +4051,14 @@ progress(failed, CaseNum, Mod, Func, GrName, Loc, {testcase_aborted,Reason}, _T,      FormatLoc = test_server_sup:format_loc(Loc),      print(minor, "=== Location: ~ts", [FormatLoc]),      print(minor, -	  escape_chars(io_lib:format("=== Reason: {testcase_aborted,~p}", +	  escape_chars(io_lib:format("=== Reason: {testcase_aborted,~tp}",  				     [Reason])),  	  []),      failed;  progress(failed, CaseNum, Mod, Func, GrName, unknown, Reason, Time,  	 Comment0, {St0,St1}) -> -    print(major, "=result        failed: ~p, ~w", [Reason,unknown_location]), +    print(major, "=result        failed: ~tp, ~w", [Reason,unknown_location]),      print(1, "*** FAILED ~ts ***",  	  [get_info_str(Mod,Func, CaseNum, get(test_server_cases))]),      test_server_sup:framework_call(report, [tc_done,{Mod,{Func,GrName}, @@ -4056,7 +4066,7 @@ progress(failed, CaseNum, Mod, Func, GrName, unknown, Reason, Time,      TimeStr = io_lib:format(if is_float(Time) -> "~.3fs";  			       true -> "~w"  			    end, [Time]), -    ErrorReason = escape_chars(lists:flatten(io_lib:format("~p", [Reason]))), +    ErrorReason = escape_chars(lists:flatten(io_lib:format("~tp", [Reason]))),      ErrorReason1 = lists:flatten([string:strip(S,left) ||  				  S <- string:tokens(ErrorReason,[$\n])]),      ErrorReason2 = @@ -4093,7 +4103,7 @@ progress(failed, CaseNum, Mod, Func, GrName, Loc, Reason, Time,  			      end;  			 true -> {Loc,Loc}  		       end, -    print(major, "=result        failed: ~p, ~p", [Reason,LocMaj]), +    print(major, "=result        failed: ~tp, ~tp", [Reason,LocMaj]),      print(1, "*** FAILED ~ts ***",  	  [get_info_str(Mod,Func, CaseNum, get(test_server_cases))]),      test_server_sup:framework_call(report, [tc_done,{Mod,{Func,GrName}, @@ -4236,7 +4246,7 @@ update_skip_counters(Pat, {US,AS}) ->      Result.  get_info_str(Mod,Func, 0, _Cases) -> -    io_lib:format("~w", [{Mod,Func}]); +    io_lib:format("~tw", [{Mod,Func}]);  get_info_str(_Mod,_Func, CaseNum, unknown) ->      "test case " ++ integer_to_list(CaseNum);  get_info_str(_Mod,_Func, CaseNum, Cases) -> @@ -4251,11 +4261,11 @@ print_if_known(Known, {SK,AK}, {SU,AU}) ->  to_string(Term) when is_list(Term) ->      case (catch io_lib:format("~ts", [Term])) of -	{'EXIT',_} -> lists:flatten(io_lib:format("~p", [Term])); +	{'EXIT',_} -> lists:flatten(io_lib:format("~tp", [Term]));  	String     -> lists:flatten(String)      end;  to_string(Term) -> -    lists:flatten(io_lib:format("~p", [Term])). +    lists:flatten(io_lib:format("~tp", [Term])).  get_last_loc(Loc) when is_tuple(Loc) ->      Loc; @@ -4327,14 +4337,14 @@ format_exception(Reason={_Error,Stack}) when is_list(Stack) ->  	undefined ->  	    case application:get_env(test_server, format_exception) of  		{ok,false} -> -		    {"~p",Reason}; +		    {"~tp",Reason};  		_ ->  		    do_format_exception(Reason)  	    end;  	FW ->  	    case application:get_env(FW, format_exception) of  		{ok,false} -> -		    {"~p",Reason}; +		    {"~tp",Reason};  		_ ->  		    do_format_exception(Reason)  	    end @@ -4345,19 +4355,19 @@ format_exception(Error) ->  do_format_exception(Reason={Error,Stack}) ->      StackFun = fun(_, _, _) -> false end,      PF = fun(Term, I) -> -		 io_lib:format("~." ++ integer_to_list(I) ++ "p", [Term]) +		 io_lib:format("~." ++ integer_to_list(I) ++ "tp", [Term])  	 end, -    case catch lib:format_exception(1, error, Error, Stack, StackFun, PF) of -	{'EXIT',_} -> -	    {"~p",Reason}; +    case catch lib:format_exception(1, error, Error, Stack, StackFun, PF, utf8) of +	{'EXIT',_R} -> +	    {"~tp",Reason};  	Formatted  -> -	    Formatted1 = re:replace(Formatted, "exception error: ", "", [{return,list}]), +	    Formatted1 = re:replace(Formatted, "exception error: ", "", [{return,list},unicode]),  	    {"~ts",lists:flatten(Formatted1)}      end.  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% -%% run_test_case_apply(Mod, Func, Args, Name, RunInit, +%% run_test_case_apply(CaseNum, Mod, Func, Args, Name, RunInit,  %%                     TimetrapData) ->  %%  {{Time,RetVal,Loc,Opts,Comment},DetectedFail,ProcessesBefore,ProcessesAfter} |  %%  {{died,Reason,unknown,Comment},DetectedFail,ProcessesBefore,ProcessesAfter} @@ -4371,9 +4381,9 @@ do_format_exception(Reason={Error,Stack}) ->  %% ProcessesBefore = ProcessesAfter = integer()  %% -run_test_case_apply(Mod, Func, Args, Name, RunInit, +run_test_case_apply(CaseNum, Mod, Func, Args, Name, RunInit,  		    TimetrapData) -> -    test_server:run_test_case_apply({Mod,Func,Args,Name,RunInit, +    test_server:run_test_case_apply({CaseNum,Mod,Func,Args,Name,RunInit,  				     TimetrapData}).  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% @@ -4457,7 +4467,7 @@ format(Detail, Format, Args) ->      Str =  	case catch io_lib:format(Format, Args) of  	    {'EXIT',_} -> -		io_lib:format("illegal format; ~p with args ~p.\n", +		io_lib:format("illegal format; ~tp with args ~tp.\n",  			      [Format,Args]);  	    Valid -> Valid  	end, @@ -4853,7 +4863,7 @@ collect_files(Dir, Pattern, St, Mode) ->      Wc = filename:join([Dir1,Pattern++"{.erl,"++code:objfile_extension()++"}"]),      case catch filelib:wildcard(Wc) of  	{'EXIT', Reason} -> -	    io:format("Could not collect files: ~p~n", [Reason]), +	    io:format("Could not collect files: ~tp~n", [Reason]),  	    {error,{collect_fail,Dir,Pattern}};  	Files ->  	    %% convert to module names and remove duplicates @@ -4897,13 +4907,13 @@ check_deny_req({Req,Val}, DenyList) ->      %%io:format("ValCheck ~p=~p in ~p\n", [Req,Val,DenyList]),      case lists:keysearch(Req, 1, DenyList) of  	{value,{_Req,DenyVal}} when Val >= DenyVal -> -	    {denied,io_lib:format("Requirement ~p=~p", [Req,Val])}; +	    {denied,io_lib:format("Requirement ~tp=~tp", [Req,Val])};  	_ ->  	    check_deny_req(Req, DenyList)      end;  check_deny_req(Req, DenyList) ->      case lists:member(Req, DenyList) of -	true -> {denied,io_lib:format("Requirement ~p", [Req])}; +	true -> {denied,io_lib:format("Requirement ~tp", [Req])};  	false -> granted      end. @@ -5004,7 +5014,7 @@ get_target_info() ->  start_node(Name, Type, Options) ->      T = 10 * ?ACCEPT_TIMEOUT * test_server:timetrap_scale_factor(), -    format(minor, "Attempt to start ~w node ~p with options ~p", +    format(minor, "Attempt to start ~w node ~tp with options ~tp",  	   [Type, Name, Options]),      case controller_call({start_node,Name,Type,Options}, T) of  	{{ok,Nodename}, Host, Cmd, Info, Warning} -> @@ -5026,16 +5036,16 @@ start_node(Name, Type, Options) ->  	{fail,{Ret, Host, Cmd}}  ->  	    format(minor,  		   "Failed to start node ~tp on ~tp with command: ~ts~n" -		   "Reason: ~p", +		   "Reason: ~tp",  		   [Name, Host, Cmd, Ret]),  	    {fail,Ret};  	{Ret, undefined, undefined} -> -	    format(minor, "Failed to start node ~tp: ~p", [Name,Ret]), +	    format(minor, "Failed to start node ~tp: ~tp", [Name,Ret]),  	    Ret;  	{Ret, Host, Cmd} ->  	    format(minor,  		   "Failed to start node ~tp on ~tp with command: ~ts~n" -		   "Reason: ~p", +		   "Reason: ~tp",  		   [Name, Host, Cmd, Ret]),  	    Ret      end. @@ -5134,8 +5144,8 @@ display_info([Pid|T], R, M) ->  	    Reds  = fetch(reductions, Info),  	    LM = length(fetch(messages, Info)),  	    pformat(io_lib:format("~w", [Pid]), -		    io_lib:format("~w", [Call]), -		    io_lib:format("~w", [Curr]), Reds, LM), +		    io_lib:format("~tw", [Call]), +		    io_lib:format("~tw", [Curr]), Reds, LM),  	    display_info(T, R+Reds, M + LM)      end;  display_info([], R, M) -> @@ -5249,11 +5259,11 @@ read_cover_file(CoverFile) ->  	    case check_cover_file(List, [], [], []) of  		{ok,Exclude,Include,Cross} -> {Exclude,Include,Cross};  		error -> -		    io:fwrite("Faulty format of CoverFile ~p\n", [CoverFile]), +		    io:fwrite("Faulty format of CoverFile ~tp\n", [CoverFile]),  		    {[],[],[]}  	    end;  	{error,Reason} -> -	    io:fwrite("Can't read CoverFile ~ts\nReason: ~p\n", +	    io:fwrite("Can't read CoverFile ~ts\nReason: ~tp\n",  		      [CoverFile,Reason]),  	    {[],[],[]}      end. @@ -5521,8 +5531,8 @@ write_coverlog_header(CoverLog) ->      case catch io:put_chars(CoverLog,html_header("Coverage results")) of  	{'EXIT',Reason} ->  	    io:format("\n\nERROR: Could not write normal heading in coverlog.\n" -		      "CoverLog: ~w\n" -		      "Reason: ~p\n", +		      "CoverLog: ~tw\n" +		      "Reason: ~tp\n",  		      [CoverLog,Reason]),  	    io:format(CoverLog,"<html><body>\n", []);  	_ -> diff --git a/lib/common_test/src/test_server_gl.erl b/lib/common_test/src/test_server_gl.erl index 4845b86dd3..ce7682d101 100644 --- a/lib/common_test/src/test_server_gl.erl +++ b/lib/common_test/src/test_server_gl.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2012-2016. All Rights Reserved. +%% Copyright Ericsson AB 2012-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -173,8 +173,8 @@ handle_info({'DOWN',Ref,process,_,Reason}=D, #st{minor_monitor=Ref}=St) ->      case Reason of  	normal -> ok;  	_ -> -	    Data = io_lib:format("=== WARNING === TC: ~w\n" -				 "Got down from minor Fd ~w: ~w\n\n", +	    Data = io_lib:format("=== WARNING === TC: ~tw\n" +				 "Got down from minor Fd ~w: ~tw\n\n",  				 [St#st.tc,St#st.minor,D]),  	    test_server_io:print_unexpected(Data)      end, @@ -319,7 +319,7 @@ output(Level, Str, Sender, From, St) when is_atom(Level) ->      output_to_file(Level, dress_output(Str, Sender, St), From, St).  output_to_file(minor, Data0, From, #st{tc={M,F,A},minor=none}) -> -    Data = [io_lib:format("=== ~w:~w/~w\n", [M,F,A]),Data0], +    Data = [io_lib:format("=== ~w:~tw/~w\n", [M,F,A]),Data0],      test_server_io:print(From, unexpected_io, Data),      ok;  output_to_file(minor, Data, From, #st{tc=TC,minor=Fd}) -> @@ -328,10 +328,10 @@ output_to_file(minor, Data, From, #st{tc=TC,minor=Fd}) ->      catch  	Type:Reason ->  	    Data1 = -		[io_lib:format("=== ERROR === TC: ~w\n" +		[io_lib:format("=== ERROR === TC: ~tw\n"  			       "Failed to write to minor Fd: ~w\n"  			       "Type: ~w\n" -			       "Reason: ~w\n", +			       "Reason: ~tw\n",  			       [TC,Fd,Type,Reason]),  		 Data,"\n"],  	    test_server_io:print(From, unexpected_io, Data1) diff --git a/lib/common_test/src/test_server_io.erl b/lib/common_test/src/test_server_io.erl index fdabf17b08..062e3bd8ff 100644 --- a/lib/common_test/src/test_server_io.erl +++ b/lib/common_test/src/test_server_io.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2012-2016. All Rights Reserved. +%% Copyright Ericsson AB 2012-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -359,7 +359,7 @@ handle_info(kill_group_leaders, #st{gls=Gls,stopping=From,  		      end, St#st{phase=idle,pending_ops=[]}, Ops),      {noreply,St1};  handle_info(Other, St) -> -    io:format("Ignoring: ~p\n", [Other]), +    io:format("Ignoring: ~tp\n", [Other]),      {noreply,St}.  terminate(_, _) -> @@ -395,7 +395,7 @@ do_output(Tag, Str, Phase, #st{fds=Fds}=St) ->  	none when Phase /= started ->  	    buffer;  	none -> -	    S = io_lib:format("\n*** ERROR: ~w, line ~w: No known '~p' log file\n", +	    S = io_lib:format("\n*** ERROR: ~w, line ~w: No known '~tp' log file\n",  			      [?MODULE,?LINE,Tag]),  	    do_output(stdout, [S,Str], Phase, St);  	{value,Fd} -> @@ -407,7 +407,7 @@ do_output(Tag, Str, Phase, #st{fds=Fds}=St) ->  		end  	    catch _:Error ->  		    S = io_lib:format("\n*** ERROR: ~w, line ~w: Error writing to " -				      "log file '~p': ~p\n", +				      "log file '~tp': ~tp\n",  				      [?MODULE,?LINE,Tag,Error]),  		    do_output(stdout, [S,Str], Phase, St)  	    end diff --git a/lib/common_test/src/test_server_node.erl b/lib/common_test/src/test_server_node.erl index 0b406c54cc..c0d7e12721 100644 --- a/lib/common_test/src/test_server_node.erl +++ b/lib/common_test/src/test_server_node.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2002-2016. All Rights Reserved. +%% Copyright Ericsson AB 2002-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -18,11 +18,11 @@  %% %CopyrightEnd%  %%  -module(test_server_node). --compile(r12). +-compile(r16).  %%%  %%% The same compiled code for this module must be possible to load -%%% in R12B and later. +%%% in R16B and later.  %%%  %% Test Controller interface @@ -237,23 +237,23 @@ print_trc(Out,{trace_ts,P,call,{M,F,A},C,Ts},N) ->      io:format(Out,  	      "~w: ~s~n"  	      "Process   : ~w~n" -	      "Call      : ~w:~w/~w~n" -	      "Arguments : ~p~n" -	      "Caller    : ~w~n~n", +	      "Call      : ~w:~tw/~w~n" +	      "Arguments : ~tp~n" +	      "Caller    : ~tw~n~n",  	      [N,ts(Ts),P,M,F,length(A),A,C]);  print_trc(Out,{trace_ts,P,call,{M,F,A},Ts},N) ->      io:format(Out,  	      "~w: ~s~n"  	      "Process   : ~w~n" -	      "Call      : ~w:~w/~w~n" -	      "Arguments : ~p~n~n", +	      "Call      : ~w:~tw/~w~n" +	      "Arguments : ~tp~n~n",  	      [N,ts(Ts),P,M,F,length(A),A]);  print_trc(Out,{trace_ts,P,return_from,{M,F,A},R,Ts},N) ->      io:format(Out,  	      "~w: ~s~n"  	      "Process      : ~w~n" -	      "Return from  : ~w:~w/~w~n" -	      "Return value : ~p~n~n", +	      "Return from  : ~w:~tw/~w~n" +	      "Return value : ~tp~n~n",  	      [N,ts(Ts),P,M,F,A,R]);  print_trc(Out,{drop,X},N) ->      io:format(Out, @@ -263,7 +263,7 @@ print_trc(Out,Trace,N) ->      Ts = element(size(Trace),Trace),      io:format(Out,  	      "~w: ~s~n" -	      "Trace        : ~p~n~n", +	      "Trace        : ~tp~n~n",  	      [N,ts(Ts),Trace]).  ts({_, _, Micro} = Now) ->      {{Y,M,D},{H,Min,S}} = calendar:now_to_local_time(Now), @@ -580,7 +580,7 @@ kill_node(SI) ->  cast_to_list(X) when is_list(X) -> X;  cast_to_list(X) when is_atom(X) -> atom_to_list(X); -cast_to_list(X) -> lists:flatten(io_lib:format("~w", [X])). +cast_to_list(X) -> lists:flatten(io_lib:format("~tw", [X])).  %%% L contains elements of the forms @@ -692,7 +692,7 @@ find_rel_suse_2(Rel, RootWc) ->      case file:list_dir(RelDir) of  	{ok,Dirs} ->  	    case lists:filter(fun(Dir) -> -				      case re:run(Dir, Pat) of +				      case re:run(Dir, Pat, [unicode]) of  					  nomatch -> false;  					  _       -> true  				      end diff --git a/lib/common_test/src/test_server_sup.erl b/lib/common_test/src/test_server_sup.erl index 6922e01fcc..21f4be22fe 100644 --- a/lib/common_test/src/test_server_sup.erl +++ b/lib/common_test/src/test_server_sup.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 1998-2016. All Rights Reserved. +%% Copyright Ericsson AB 1998-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -83,7 +83,7 @@ kill_the_process(Pid, Timeout0, TruncTO, ReportTVal) ->  			    "Testcase process ~w not "  			    "responding to timetrap "  			    "timeout:~n" -			    "  ~p.~n" +			    "  ~tp.~n"  			    "Killing testcase...~n",  			    [Pid, Trap]),  		    exit(Pid, kill) @@ -144,11 +144,11 @@ call_crash(Time,Crash,M,F,A) ->  	    {'EXIT',Pid,_Reason} when Crash==any ->  		ok;  	    {'EXIT',Reason} -> -		test_server:format(12, "Wrong crash reason. Wanted ~p, got ~p.", +		test_server:format(12, "Wrong crash reason. Wanted ~tp, got ~tp.",  		      [Crash, Reason]),  		exit({wrong_crash_reason,Reason});  	    {'EXIT',Pid,Reason} -> -		test_server:format(12, "Wrong crash reason. Wanted ~p, got ~p.", +		test_server:format(12, "Wrong crash reason. Wanted ~tp, got ~tp.",  		      [Crash, Reason]),  		exit({wrong_crash_reason,Reason});  	    {'EXIT',OtherPid,Reason} when OldTrapExit == false -> @@ -334,11 +334,11 @@ do_appup_tests(_, _Application, Up, Down, Modules) ->                  ok ->                      test_server:format(minor, "OK~n");                  Error -> -                    test_server:format(minor, "ERROR ~p~n", [Error]), +                    test_server:format(minor, "ERROR ~tp~n", [Error]),                      test_server:fail(Error)              end;          Error -> -            test_server:format(minor, "ERROR ~p~n", [Error]), +            test_server:format(minor, "ERROR ~tp~n", [Error]),              test_server:fail(Error)      end. @@ -346,7 +346,7 @@ check_appup_clauses_plausible([], _Direction, _Modules) ->      ok;  check_appup_clauses_plausible([{Re, Instrs} | Rest], Direction, Modules)    when is_binary(Re) -> -    case re:compile(Re) of +    case re:compile(Re,[unicode]) of          {ok, _} ->              case check_appup_instructions(Instrs, Direction, Modules) of                  ok -> @@ -557,7 +557,7 @@ check_dict(Dict, Reason) ->  	[] ->  	    1;                         % All ok.  	List -> -	    io:format("** ~ts (~ts) ->~n~p~n",[Reason, Dict, List]), +	    io:format("** ~ts (~ts) ->~n~tp~n",[Reason, Dict, List]),  	    0      end. @@ -566,7 +566,7 @@ check_dict_tolerant(Dict, Reason, Mode) ->  	[] ->  	    1;                         % All ok.  	List -> -	    io:format("** ~ts (~ts) ->~n~p~n",[Reason, Dict, List]), +	    io:format("** ~ts (~ts) ->~n~tp~n",[Reason, Dict, List]),  	    case Mode of  		pedantic ->  		    0; @@ -646,7 +646,7 @@ append_files_to_logfile([File|Files]) ->  		    %% fail, but in that case it will throw an exception so that  		    %% we will be aware of the problem.  		    io:format(Fd, "Unable to write the crash dump " -			      "to this file: ~p~n", [file:format_error(Error)]) +			      "to this file: ~tp~n", [file:format_error(Error)])  	    end;  	_Error ->  	    io:format(Fd, "Failed to read: ~ts\n", [File]) @@ -802,9 +802,9 @@ format_loc([{Mod,Func,Line}|Rest]) ->  format_loc([{Mod,LineOrFunc}]) ->      format_loc({Mod,LineOrFunc});  format_loc({Mod,Func}) when is_atom(Func) ->  -    io_lib:format("{~w,~w}",[Mod,Func]); +    io_lib:format("{~w,~tw}",[Mod,Func]);  format_loc(Loc) -> -    io_lib:format("~p",[Loc]). +    io_lib:format("~tp",[Loc]).  format_loc1([{Mod,Func,Line}]) ->      ["              ",format_loc1({Mod,Func,Line}),"]"]; @@ -824,12 +824,12 @@ format_loc1({Mod,Func,Line}) ->  		      true ->  			   Line  		   end, -	    io_lib:format("{~w,~w,<a href=\"~ts~ts#~s\">~w</a>}", +	    io_lib:format("{~w,~tw,<a href=\"~ts~ts#~ts\">~tw</a>}",  			  [Mod,Func,  			   test_server_ctrl:uri_encode(downcase(ModStr)),  			   ?src_listing_ext,Link,Line]);  	_ -> -	    io_lib:format("{~w,~w,~w}",[Mod,Func,Line]) +	    io_lib:format("{~w,~tw,~tw}",[Mod,Func,Line])      end.  downcase(S) -> downcase(S, []). diff --git a/lib/common_test/src/unix_telnet.erl b/lib/common_test/src/unix_telnet.erl index 4897ddb2f8..8ac467014c 100644 --- a/lib/common_test/src/unix_telnet.erl +++ b/lib/common_test/src/unix_telnet.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2004-2016. All Rights Reserved. +%% Copyright Ericsson AB 2004-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -53,8 +53,6 @@  %%% @see ct_telnet  -module(unix_telnet). --compile(export_all). -  %% Callbacks for ct_telnet.erl  -export([connect/7,get_prompt_regexp/0]).  -import(ct_telnet,[start_gen_log/1,log/4,end_gen_log/0]). @@ -134,25 +132,25 @@ connect1(Name,Ip,Port,Timeout,KeepAlive,TCPNoDelay,Username,Password) ->  					 Prompt=/=?password ->  					{ok,Pid};  				    Error -> -					log(Name,recv,"Password failed\n~p\n", +					log(Name,recv,"Password failed\n~tp\n",  					    [Error]),  					{error,Error}  				end;  			    Error -> -				log(Name,recv,"Login to ~p:~p failed\n~p\n",[Ip,Port,Error]), +				log(Name,recv,"Login to ~p:~p failed\n~tp\n",[Ip,Port,Error]),  				{error,Error}  			end;  		    {ok,[{prompt,_OtherPrompt1},{prompt,_OtherPrompt2}],_} ->  			{ok,Pid};  		    Error ->  			log(Name,conn_error, -			    "Did not get expected prompt from ~p:~p\n~p\n", +			    "Did not get expected prompt from ~p:~p\n~tp\n",  			    [Ip,Port,Error]),  			{error,Error}  		end;  	    Error ->  		log(Name,conn_error, -		    "Could not open telnet connection to ~p:~p\n~p\n", +		    "Could not open telnet connection to ~p:~p\n~tp\n",  		    [Ip,Port,Error]),  		Error  	end, diff --git a/lib/common_test/src/vts.erl b/lib/common_test/src/vts.erl index f1c5051164..99a109cfe8 100644 --- a/lib/common_test/src/vts.erl +++ b/lib/common_test/src/vts.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %% -%% Copyright Ericsson AB 2003-2016. All Rights Reserved. +%% Copyright Ericsson AB 2003-2017. All Rights Reserved.  %%  %% Licensed under the Apache License, Version 2.0 (the "License");  %% you may not use this file except in compliance with the License. @@ -250,7 +250,7 @@ loop(State) ->  	{'EXIT',Pid,Reason} ->  	    case State#state.test_runner of  		Pid -> -		    io:format("Test run error: ~p\n",[Reason]), +		    io:format("Test run error: ~tp\n",[Reason]),  		    loop(State);  		_ ->  		    loop(State) @@ -551,7 +551,7 @@ case_select(Dir,Suite,Case,N) ->  	    true = code:add_pathz(Dir),  	    case catch apply(Suite,all,[]) of  		{'EXIT',Reason} -> -		    io:format("\n~p\n",[Reason]), +		    io:format("\n~tp\n",[Reason]),  		    red(["COULD NOT READ TESTCASES!!",br(),  			 "See erlang shell for info"]);  		{skip,_Reason} -> | 
