diff options
Diffstat (limited to 'lib')
31 files changed, 1230 insertions, 257 deletions
| diff --git a/lib/common_test/doc/src/Makefile b/lib/common_test/doc/src/Makefile index e495f587a3..152ece5d25 100644 --- a/lib/common_test/doc/src/Makefile +++ b/lib/common_test/doc/src/Makefile @@ -53,7 +53,8 @@ XML_REF3_FILES = ct.xml \  	ct_slave.xml \  	ct_property_test.xml \  	ct_netconfc.xml \ -	ct_hooks.xml +	ct_hooks.xml \ +        ct_testspec.xml  XML_REF6_FILES = common_test_app.xml  XML_PART_FILES = part.xml  diff --git a/lib/common_test/doc/src/ct_testspec.xml b/lib/common_test/doc/src/ct_testspec.xml new file mode 100644 index 0000000000..36893f66cf --- /dev/null +++ b/lib/common_test/doc/src/ct_testspec.xml @@ -0,0 +1,84 @@ +<?xml version="1.0" encoding="utf-8" ?> +<!DOCTYPE erlref SYSTEM "erlref.dtd"> + +<erlref> +  <header> +    <copyright> +      <year>2016</year> +      <holder>Ericsson AB. All Rights Reserved.</holder> +    </copyright> +    <legalnotice> +      Licensed under the Apache License, Version 2.0 (the "License"); +      you may not use this file except in compliance with the License. +      You may obtain a copy of the License at + +          http://www.apache.org/licenses/LICENSE-2.0 + +      Unless required by applicable law or agreed to in writing, software +      distributed under the License is distributed on an "AS IS" BASIS, +      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +      See the License for the specific language governing permissions and +      limitations under the License. + +    </legalnotice> + +    <title>ct_testspec</title> +    <prepared></prepared> +    <responsible></responsible> +    <docno></docno> +    <approved></approved> +    <checked></checked> +    <date></date> +    <rev>A</rev> +    <file>ct_testspec.xml</file> +  </header> +  <module>ct_testspec</module> +  <modulesummary>Parsing of test specifications for Common Test. +  </modulesummary> + +<description> + +    <p>Parsing of test specifications for <c>Common Test</c>.</p> + +    <p>This module exports help functions for parsing of test specifications.</p> + +</description> + +  <funcs> +    <func> +    <name>get_tests(SpecsIn) -> {ok, [{Specs,Tests}]} | {error, Reason}</name> +    <fsummary>Parse the given test specification files and return the tests to run and skip.</fsummary> +      <type> +        <v>SpecsIn = [string()] | [[string()]]</v> +        <v>Specs = [string()]</v> +        <v>Test = [{Node,Run,Skip}]</v> +        <v>Node = atom()</v> +        <v>Run = {Dir,Suites,Cases}</v> +        <v>Skip = {Dir,Suites,Comment} | {Dir,Suites,Cases,Comment}</v> +        <v>Dir = string()</v> +        <v>Suites = atom | [atom()] | all</v> +        <v>Cases = atom | [atom()] | all</v> +        <v>Comment = string()</v> +        <v>Reason = term()</v> +      </type> +      <desc><marker id="add_nodes-1"/> +        <p>Parse the given test specification files and return the +        tests to run and skip.</p> + +        <p>If <c>SpecsIn=[Spec1,Spec2,...]</c>, separate tests will be +        created per specification. If +        <c>SpecsIn=[[Spec1,Spec2,...]]</c>, all specifications will be +        merge into one test.</p> + +        <p>For each test, a <c>{Specs,Tests}</c> element is returned, +        where <c>Specs</c> is a list of all included test +        specifications, and <c>Tests</c> specifies actual tests to +        run/skip per node.</p> +      </desc> +    </func> + +  </funcs> + +</erlref> + + diff --git a/lib/common_test/doc/src/ref_man.xml b/lib/common_test/doc/src/ref_man.xml index d1567e2d3c..1ac20db5c2 100644 --- a/lib/common_test/doc/src/ref_man.xml +++ b/lib/common_test/doc/src/ref_man.xml @@ -47,6 +47,7 @@    <xi:include href="ct_slave.xml"/>    <xi:include href="ct_hooks.xml"/>    <xi:include href="ct_property_test.xml"/> +  <xi:include href="ct_testspec.xml"/>  </application> diff --git a/lib/common_test/src/ct_testspec.erl b/lib/common_test/src/ct_testspec.erl index 991abb0666..16001ce4c8 100644 --- a/lib/common_test/src/ct_testspec.erl +++ b/lib/common_test/src/ct_testspec.erl @@ -26,7 +26,8 @@  -export([prepare_tests/1, prepare_tests/2,   	 collect_tests_from_list/2, collect_tests_from_list/3, -	 collect_tests_from_file/2, collect_tests_from_file/3]). +	 collect_tests_from_file/2, collect_tests_from_file/3, +         get_tests/1]).  -export([testspec_rec2list/1, testspec_rec2list/2]). @@ -803,6 +804,31 @@ list_nodes(#testspec{nodes=NodeRefs}) ->      lists:map(fun({_Ref,Node}) -> Node end, NodeRefs).		       +%%%----------------------------------------------------------------- +%%% Parse the given test specs and return the complete set of specs +%%% and tests to run/skip. +%%% [Spec1,Spec2,...] means create separate tests per spec +%%% [[Spec1,Spec2,...]] means merge all specs into one +-spec get_tests(Specs) -> {ok,[{Specs,Tests}]} | {error,Reason} when +      Specs :: [string()] | [[string()]], +      Tests :: {Node,Run,Skip}, +      Node :: atom(), +      Run :: {Dir,Suites,Cases}, +      Skip :: {Dir,Suites,Comment} | {Dir,Suites,Cases,Comment}, +      Dir :: string(), +      Suites :: atom | [atom()] | all, +      Cases :: atom | [atom()] | all, +      Comment :: string(), +      Reason :: term(). + +get_tests(Specs) -> +    case collect_tests_from_file(Specs,true) of +        Tests when is_list(Tests) -> +            {ok,[{S,prepare_tests(R)} || {S,R} <- Tests]}; +        Error -> +            Error +    end. +  %%     -----------------------------------------------------  %%   /                                                       \  %%  |  When adding test/config terms, remember to update      | diff --git a/lib/common_test/test/ct_testspec_2_SUITE.erl b/lib/common_test/test/ct_testspec_2_SUITE.erl index 1a941df185..1bab80942a 100644 --- a/lib/common_test/test/ct_testspec_2_SUITE.erl +++ b/lib/common_test/test/ct_testspec_2_SUITE.erl @@ -220,7 +220,24 @@ basic_compatible_no_nodes(_Config) ->  					    {tc2,{skip,"skipped"}}]}]}],  		       merge_tests = true}, -    verify_result(Verify,ListResult,FileResult). +    verify_result(Verify,ListResult,FileResult), + +    {ok,Tests} = ct_testspec:get_tests([SpecFile]), +    ct:pal("ct_testspec:get_tests/1:~n~p~n", [Tests]), +    [{[SpecFile],[{Node,Run,Skip}]}] = Tests, +    [{Alias1V,x_SUITE,all}, +     {Alias1V,y_SUITE,[{g1,all},{g2,all},tc1,tc2]}, +     {Alias1V,z_SUITE,all}, +     {Alias2V,x_SUITE,all}, +     {Alias2V,y_SUITE,all}] = lists:sort(Run), +    [{Alias1V,z_SUITE,"skipped"}, +     {Alias2V,x_SUITE,{g1,all},"skipped"}, +     {Alias2V,x_SUITE,{g2,all},"skipped"}, +     {Alias2V,y_SUITE,tc1,"skipped"}, +     {Alias2V,y_SUITE,tc2,"skipped"}] = lists:sort(Skip), + +    ok. +  %%%-----------------------------------------------------------------  %%% @@ -346,7 +363,25 @@ basic_compatible_nodes(_Config) ->  					    {tc2,{skip,"skipped"}}]}]}],  		       merge_tests = true}, -    verify_result(Verify,ListResult,FileResult). +    verify_result(Verify,ListResult,FileResult), + +    {ok,Tests} = ct_testspec:get_tests([SpecFile]), +    ct:pal("ct_testspec:get_tests/1:~n~p~n", [Tests]), +    [{[SpecFile],[{Node,[],[]}, +                  {Node1,Run1,Skip1}, +                  {Node2,Run2,Skip2}]}] = Tests, +    [{TO1V,x_SUITE,all}, +     {TO1V,y_SUITE,[{g1,all},{g2,all},tc1,tc2]}, +     {TO1V,z_SUITE,all}] = lists:sort(Run1), +    [{TO2V,x_SUITE,all}, +     {TO2V,y_SUITE,all}] = lists:sort(Run2), +    [{TO1V,z_SUITE,"skipped"}] = lists:sort(Skip1), +    [{TO2V,x_SUITE,{g1,all},"skipped"}, +     {TO2V,x_SUITE,{g2,all},"skipped"}, +     {TO2V,y_SUITE,tc1,"skipped"}, +     {TO2V,y_SUITE,tc2,"skipped"}] = lists:sort(Skip2), + +    ok.  %%%-----------------------------------------------------------------  %%% @@ -439,7 +474,28 @@ no_merging(_Config) ->  				 [{y_SUITE,[{tc1,{skip,"skipped"}},  					    {tc2,{skip,"skipped"}}]}]}]}, -    verify_result(Verify,ListResult,FileResult). +    verify_result(Verify,ListResult,FileResult), + +    {ok,Tests} = ct_testspec:get_tests([SpecFile]), +    ct:pal("ct_testspec:get_tests/1:~n~p~n", [Tests]), +    [{[SpecFile],[{Node,[],[]}, +                  {Node1,Run1,Skip1}, +                  {Node2,Run2,Skip2}]}] = Tests, +    [{TO1V,x_SUITE,all}, +     {TO1V,y_SUITE,[tc1,tc2]}, +     {TO1V,y_SUITE,[{g1,all},{g2,all}]}, +     {TO1V,z_SUITE,all}] = lists:sort(Run1), +    [{TO2V,x_SUITE,all}, +     {TO2V,x_SUITE,[{skipped,g1,all},{skipped,g2,all}]}, +     {TO2V,y_SUITE,all}, +     {TO2V,y_SUITE,[{skipped,tc1},{skipped,tc2}]}] = lists:sort(Run2), +    [{TO1V,z_SUITE,"skipped"}] = lists:sort(Skip1), +    [{TO2V,x_SUITE,{g1,all},"skipped"}, +     {TO2V,x_SUITE,{g2,all},"skipped"}, +     {TO2V,y_SUITE,tc1,"skipped"}, +     {TO2V,y_SUITE,tc2,"skipped"}] = lists:sort(Skip2), + +    ok.  %%%-----------------------------------------------------------------  %%% @@ -510,7 +566,25 @@ multiple_specs(_Config) ->  				  {y_SUITE,[all,{tc1,{skip,"skipped"}},  					    {tc2,{skip,"skipped"}}]}]}]}, -    verify_result(Verify,FileResult,FileResult). +    verify_result(Verify,FileResult,FileResult), + +    {ok,Tests} = ct_testspec:get_tests([[SpecFile1,SpecFile2]]), +    ct:pal("ct_testspec:get_tests/1:~n~p~n", [Tests]), +    [{[SpecFile1,SpecFile2],[{Node,[],[]}, +                             {Node1,Run1,Skip1}, +                             {Node2,Run2,Skip2}]}] = Tests, +    [{TO1V,x_SUITE,all}, +     {TO1V,y_SUITE,[{g1,all},{g2,all},tc1,tc2]}, +     {TO1V,z_SUITE,all}] = lists:sort(Run1), +    [{TO2V,x_SUITE,all}, +     {TO2V,y_SUITE,all}] = lists:sort(Run2), +    [{TO1V,z_SUITE,"skipped"}] = lists:sort(Skip1), +    [{TO2V,x_SUITE,{g1,all},"skipped"}, +     {TO2V,x_SUITE,{g2,all},"skipped"}, +     {TO2V,y_SUITE,tc1,"skipped"}, +     {TO2V,y_SUITE,tc2,"skipped"}] = lists:sort(Skip2), + +    ok.  %%%-----------------------------------------------------------------  %%%  diff --git a/lib/compiler/doc/src/compile.xml b/lib/compiler/doc/src/compile.xml index 3ce37b98e9..611cdc71ae 100644 --- a/lib/compiler/doc/src/compile.xml +++ b/lib/compiler/doc/src/compile.xml @@ -669,7 +669,7 @@ module.beam: module.erl \        <fsummary>Compiles a list of forms.</fsummary>        <desc>          <p>Is the same as -	  <c>forms(File, [verbose,report_errors,report_warnings])</c>. +	  <c>forms(Forms, [verbose,report_errors,report_warnings])</c>.  	</p>        </desc>      </func> diff --git a/lib/erl_interface/doc/src/erl_call.xml b/lib/erl_interface/doc/src/erl_call.xml index f1e52b1889..426f6b88ca 100644 --- a/lib/erl_interface/doc/src/erl_call.xml +++ b/lib/erl_interface/doc/src/erl_call.xml @@ -193,7 +193,7 @@ erl_call -s -a 'erlang halt' -n madonna      <p>To apply with many arguments:</p>      <code type="none"><![CDATA[ -erl_call -s -a 'lists map [{math,sqrt},[1,4,9,16,25]]' -n madonna +erl_call -s -a 'lists seq [1,10]' -n madonna      ]]></code>      <p>To evaluate some expressions diff --git a/lib/kernel/src/os.erl b/lib/kernel/src/os.erl index f8519d3a5e..03d4324992 100644 --- a/lib/kernel/src/os.erl +++ b/lib/kernel/src/os.erl @@ -289,12 +289,11 @@ get_data(Port, MonRef, Eot, Sofar) ->                  more ->                      get_data(Port, MonRef, Eot, [Sofar,Bytes]);                  Last -> -                    Port ! {self(), close}, -                    flush_until_closed(Port), -                    flush_exit(Port), +                    catch port_close(Port), +                    flush_until_down(Port, MonRef),                      iolist_to_binary([Sofar, Last])              end; -        {'DOWN', MonRef, _, _ , _} -> +        {'DOWN', MonRef, _, _, _} ->  	    flush_exit(Port),  	    iolist_to_binary(Sofar)      end. @@ -308,18 +307,25 @@ eot(Bs, Eot) ->              binary:part(Bs,{0, Pos})      end. -flush_until_closed(Port) -> +%% When port_close returns we know that all the +%% messages sent have been sent and that the +%% DOWN message is after them all. +flush_until_down(Port, MonRef) ->      receive          {Port, {data, _Bytes}} -> -            flush_until_closed(Port); -        {Port, closed} -> -            true +            flush_until_down(Port, MonRef); +        {'DOWN', MonRef, _, _, _} -> +            flush_exit(Port)      end. +%% The exit signal is always delivered before +%% the down signal, so we can be sure that if there +%% was an exit message sent, it will be in the +%% mailbox now.  flush_exit(Port) ->      receive          {'EXIT',  Port,  _} ->              ok -    after 1 ->				% force context switch +    after 0 ->              ok      end. diff --git a/lib/kernel/test/code_SUITE.erl b/lib/kernel/test/code_SUITE.erl index c5167efa56..5777b397b8 100644 --- a/lib/kernel/test/code_SUITE.erl +++ b/lib/kernel/test/code_SUITE.erl @@ -588,20 +588,28 @@ sticky_compiler(Files, PrivDir) ->      [R || R <- Rets, R =/= ok].  do_sticky_compile(Mod, Dir) -> -    %% Make sure that the module is loaded. A module being sticky -    %% only prevents it from begin reloaded, not from being loaded -    %% from the wrong place to begin with. -    Mod = Mod:module_info(module), -    File = filename:append(Dir, atom_to_list(Mod)), -    Src = io_lib:format("-module(~s).\n" -			"-export([test/1]).\n" -			"test(me) -> fail.\n", [Mod]), -    ok = file:write_file(File++".erl", Src), -    case c:c(File, [{outdir,Dir}]) of -	{ok,Module} -> -	    Module:test(me); -	{error,sticky_directory} -> -	    ok +    case code:is_sticky(Mod) of +        true -> +            %% Make sure that the module is loaded. A module being sticky +            %% only prevents it from begin reloaded, not from being loaded +            %% from the wrong place to begin with. +            Mod = Mod:module_info(module), +            File = filename:append(Dir, atom_to_list(Mod)), +            Src = io_lib:format("-module(~s).\n" +                                "-export([test/1]).\n" +                                "test(me) -> fail.\n", [Mod]), +            ok = file:write_file(File++".erl", Src), +            case c:c(File, [{outdir,Dir}]) of +                {ok,Module} -> +                    Module:test(me); +                {error,sticky_directory} -> +                    ok +            end; +        false -> +            %% For some reason the module is not sticky +            %% could be that the .erlang file has +            %% unstuck it? +            {Mod, is_not_sticky}      end.  %% Test that the -pa and -pz options work as expected. diff --git a/lib/ssh/doc/src/using_ssh.xml b/lib/ssh/doc/src/using_ssh.xml index 0861c641c7..864378b640 100644 --- a/lib/ssh/doc/src/using_ssh.xml +++ b/lib/ssh/doc/src/using_ssh.xml @@ -305,7 +305,7 @@ ok = erl_tar:close(HandleRead),      <code type="erl" >  -module(ssh_echo_server). --behaviour(ssh_subsystem). +-behaviour(ssh_daemon_channel).  -record(state, {  	  n,  	  id, diff --git a/lib/ssl/doc/src/ssl_session_cache_api.xml b/lib/ssl/doc/src/ssl_session_cache_api.xml index b85d8fb284..1b41eae89d 100644 --- a/lib/ssl/doc/src/ssl_session_cache_api.xml +++ b/lib/ssl/doc/src/ssl_session_cache_api.xml @@ -11,7 +11,7 @@        Licensed under the Apache License, Version 2.0 (the "License");        you may not use this file except in compliance with the License.        You may obtain a copy of the License at -  +            http://www.apache.org/licenses/LICENSE-2.0        Unless required by applicable law or agreed to in writing, software @@ -62,8 +62,8 @@      </taglist>    </section> -   -  <funcs>    + +  <funcs>      <func>        <name>delete(Cache, Key) -> _</name> @@ -134,7 +134,7 @@           </p>        </desc>      </func> -     +      <func>        <name>select_session(Cache, PartialKey) -> [session()]</name>        <fsummary>Selects sessions that can be reused.</fsummary> @@ -151,6 +151,21 @@      </func>      <func> +      <name>size(Cache) -> integer()</name> +      <fsummary>Returns the number of sessions in the cache.</fsummary> +      <type> +	<v>Cache = cache_ref()</v> +      </type> +      <desc> +	<p>Returns the number of sessions in the cache. If size +	exceeds the maximum number of sessions, the current cache +	entries will be invalidated regardless of their remaining +	lifetime. Is to be callable from any process. +	</p> +      </desc> +    </func> + +    <func>        <name>terminate(Cache) -> _</name>        <fsummary>Called by the process that handles the cache when it        is about to terminate.</fsummary> @@ -178,7 +193,7 @@  	</p>        </desc>      </func> -     -  </funcs>  -   + +  </funcs> +  </erlref> diff --git a/lib/ssl/src/tls_v1.erl b/lib/ssl/src/tls_v1.erl index 7f24ce5192..32bcdf71c3 100644 --- a/lib/ssl/src/tls_v1.erl +++ b/lib/ssl/src/tls_v1.erl @@ -204,21 +204,21 @@ suites(Minor) when Minor == 1; Minor == 2 ->        ?TLS_ECDH_RSA_WITH_AES_256_CBC_SHA,        ?TLS_RSA_WITH_AES_256_CBC_SHA, -      ?TLS_ECDHE_ECDSA_WITH_3DES_EDE_CBC_SHA, -      ?TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA, -      ?TLS_DHE_RSA_WITH_3DES_EDE_CBC_SHA, -      ?TLS_DHE_DSS_WITH_3DES_EDE_CBC_SHA, -      ?TLS_ECDH_ECDSA_WITH_3DES_EDE_CBC_SHA, -      ?TLS_ECDH_RSA_WITH_3DES_EDE_CBC_SHA, -      ?TLS_RSA_WITH_3DES_EDE_CBC_SHA, -        ?TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA,        ?TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA,        ?TLS_DHE_RSA_WITH_AES_128_CBC_SHA,        ?TLS_DHE_DSS_WITH_AES_128_CBC_SHA,        ?TLS_ECDH_ECDSA_WITH_AES_128_CBC_SHA,        ?TLS_ECDH_RSA_WITH_AES_128_CBC_SHA, -      ?TLS_RSA_WITH_AES_128_CBC_SHA +      ?TLS_RSA_WITH_AES_128_CBC_SHA, + +      ?TLS_ECDHE_ECDSA_WITH_3DES_EDE_CBC_SHA, +      ?TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA, +      ?TLS_DHE_RSA_WITH_3DES_EDE_CBC_SHA, +      ?TLS_DHE_DSS_WITH_3DES_EDE_CBC_SHA, +      ?TLS_ECDH_ECDSA_WITH_3DES_EDE_CBC_SHA, +      ?TLS_ECDH_RSA_WITH_3DES_EDE_CBC_SHA, +      ?TLS_RSA_WITH_3DES_EDE_CBC_SHA      ];  suites(3) ->      [ diff --git a/lib/stdlib/doc/src/filename.xml b/lib/stdlib/doc/src/filename.xml index 2a413835d0..f7f3f7b504 100644 --- a/lib/stdlib/doc/src/filename.xml +++ b/lib/stdlib/doc/src/filename.xml @@ -511,6 +511,33 @@ true      </func>      <func> +      <name name="safe_relative_path" arity="1"/> +      <fsummary>Sanitize a relative path to avoid directory traversal attacks.</fsummary> +      <desc> +        <p>Sanitizes the relative path by eliminating ".." and "." +        components to protect against directory traversal attacks. +        Either returns the sanitized path name, or the atom +        <c>unsafe</c> if the path is unsafe. +        The path is considered unsafe in the following circumstances:</p> +        <list type="bulleted"> +          <item><p>The path is not relative.</p></item> +          <item><p>A ".." component would climb up above the root of +          the relative path.</p></item> +        </list> +        <p><em>Examples:</em></p> +        <pre> +1> <input>filename:safe_relative_path("dir/sub_dir/..").</input> +"dir" +2> <input>filename:safe_relative_path("dir/..").</input> +[] +3> <input>filename:safe_relative_path("dir/../..").</input> +unsafe +4> <input>filename:safe_relative_path("/abs/path").</input> +unsafe</pre> +       </desc> +    </func> + +    <func>        <name name="split" arity="1"/>        <fsummary>Split a filename into its path components.</fsummary>        <desc> diff --git a/lib/stdlib/src/filename.erl b/lib/stdlib/src/filename.erl index c4586171ca..5d60b3837e 100644 --- a/lib/stdlib/src/filename.erl +++ b/lib/stdlib/src/filename.erl @@ -34,7 +34,8 @@  -export([absname/1, absname/2, absname_join/2,   	 basename/1, basename/2, dirname/1,  	 extension/1, join/1, join/2, pathtype/1, -	 rootname/1, rootname/2, split/1, nativename/1]). +         rootname/1, rootname/2, split/1, nativename/1, +         safe_relative_path/1]).  -export([find_src/1, find_src/2, flatten/1]).  -export([basedir/2, basedir/3]). @@ -750,6 +751,39 @@ separators() ->  	_ -> {false, false}      end. +-spec safe_relative_path(Filename) -> 'unsafe' | SafeFilename when +      Filename :: file:name_all(), +      SafeFilename :: file:name_all(). + +safe_relative_path(Path) -> +    case pathtype(Path) of +        relative -> +            Cs0 = split(Path), +            safe_relative_path_1(Cs0, []); +        _ -> +            unsafe +    end. + +safe_relative_path_1(["."|T], Acc) -> +    safe_relative_path_1(T, Acc); +safe_relative_path_1([<<".">>|T], Acc) -> +    safe_relative_path_1(T, Acc); +safe_relative_path_1([".."|T], Acc) -> +    climb(T, Acc); +safe_relative_path_1([<<"..">>|T], Acc) -> +    climb(T, Acc); +safe_relative_path_1([H|T], Acc) -> +    safe_relative_path_1(T, [H|Acc]); +safe_relative_path_1([], []) -> +    []; +safe_relative_path_1([], Acc) -> +    join(lists:reverse(Acc)). + +climb(_, []) -> +    unsafe; +climb(T, [_|Acc]) -> +    safe_relative_path_1(T, Acc). +  %% find_src(Module) -- diff --git a/lib/stdlib/src/io_lib_pretty.erl b/lib/stdlib/src/io_lib_pretty.erl index 16ca2f41dc..ba2cffdcb3 100644 --- a/lib/stdlib/src/io_lib_pretty.erl +++ b/lib/stdlib/src/io_lib_pretty.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 1996-2016. All Rights Reserved. +%% Copyright Ericsson AB 1996-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. @@ -128,6 +128,10 @@ max_cs(M, _Len) ->      M.  -define(ATM(T), is_list(element(1, T))). +-define(ATM_PAIR(Pair), +        ?ATM(element(2, element(1, Pair))) % Key +        andalso +        ?ATM(element(3, element(1, Pair)))). % Value  -define(ATM_FLD(Field), ?ATM(element(4, element(1, Field)))).  pp({_S, Len} = If, Col, Ll, M, _TInd, _Ind, LD, W)  @@ -140,9 +144,8 @@ pp({{tuple,true,L}, _Len}, Col, Ll, M, TInd, Ind, LD, W) ->  pp({{tuple,false,L}, _Len}, Col, Ll, M, TInd, Ind, LD, W) ->      [${, pp_list(L, Col + 1, Ll, M, TInd, indent(1, Ind), LD, $,, W + 1), $}];  pp({{map,Pairs},_Len}, Col, Ll, M, TInd, Ind, LD, W) -> -    [$#,${, pp_list(Pairs, Col + 2, Ll, M, TInd, indent(2, Ind), LD, $,, W + 1), $}]; -pp({{map_pair,K,V},_Len}, Col, Ll, M, TInd, Ind, LD, W) -> -    [pp(K, Col, Ll, M, TInd, Ind, LD, W), " => ", pp(V, Col, Ll, M, TInd, Ind, LD, W)]; +    [$#, ${, pp_map(Pairs, Col + 2, Ll, M, TInd, indent(2, Ind), LD, W + 1), +     $}];  pp({{record,[{Name,NLen} | L]}, _Len}, Col, Ll, M, TInd, Ind, LD, W) ->      [Name, ${, pp_record(L, NLen, Col, Ll, M, TInd, Ind, LD, W + NLen+1), $}];  pp({{bin,S}, _Len}, Col, Ll, M, _TInd, Ind, LD, W) -> @@ -166,6 +169,46 @@ pp_tag_tuple([{Tag,Tlen} | L], Col, Ll, M, TInd, Ind, LD, W) ->              [Tag, S | pp_list(L, Tcol, Ll, M, TInd, Indent, LD, S, W+Tlen+1)]      end. +pp_map([], _Col, _Ll, _M, _TInd, _Ind, _LD, _W) -> +    ""; +pp_map({dots, _}, _Col, _Ll, _M, _TInd, _Ind, _LD, _W) -> +    "..."; +pp_map([P | Ps], Col, Ll, M, TInd, Ind, LD, W) -> +    {PS, PW} = pp_pair(P, Col, Ll, M, TInd, Ind, last_depth(Ps, LD), W), +    [PS | pp_pairs_tail(Ps, Col, Col + PW, Ll, M, TInd, Ind, LD, PW)]. + +pp_pairs_tail([], _Col0, _Col, _Ll, _M, _TInd, _Ind, _LD, _W) -> +    ""; +pp_pairs_tail({dots, _}, _Col0, _Col, _M, _Ll, _TInd, _Ind, _LD, _W) -> +    ",..."; +pp_pairs_tail([{_, Len}=P | Ps], Col0, Col, Ll, M, TInd, Ind, LD, W) -> +    LD1 = last_depth(Ps, LD), +    ELen = 1 + Len, +    if +        LD1 =:= 0, ELen + 1 < Ll - Col, W + ELen + 1 =< M, ?ATM_PAIR(P); +        LD1 > 0, ELen < Ll - Col - LD1, W + ELen + LD1 =< M, ?ATM_PAIR(P) -> +            [$,, write_pair(P) | +             pp_pairs_tail(Ps, Col0, Col+ELen, Ll, M, TInd, Ind, LD, W+ELen)]; +        true -> +            {PS, PW} = pp_pair(P, Col0, Ll, M, TInd, Ind, LD1, 0), +            [$,, $\n, Ind, PS | +             pp_pairs_tail(Ps, Col0, Col0 + PW, Ll, M, TInd, Ind, LD, PW)] +    end. + +pp_pair({_, Len}=Pair, Col, Ll, M, _TInd, _Ind, LD, W) +         when Len < Ll - Col - LD, Len + W + LD =< M -> +    {write_pair(Pair), if +                          ?ATM_PAIR(Pair) -> +                              Len; +                          true -> +                              Ll % force nl +                      end}; +pp_pair({{map_pair, K, V}, _Len}, Col0, Ll, M, TInd, Ind0, LD, W) -> +    I = map_value_indent(TInd), +    Ind = indent(I, Ind0), +    {[pp(K, Col0, Ll, M, TInd, Ind0, LD, W), " =>\n", +      Ind | pp(V, Col0 + I, Ll, M, TInd, Ind, LD, 0)], Ll}. % force nl +  pp_record([], _Nlen, _Col, _Ll, _M, _TInd, _Ind, _LD, _W) ->      "";  pp_record({dots, _}, _Nlen, _Col, _Ll, _M, _TInd, _Ind, _LD, _W) -> @@ -204,7 +247,11 @@ pp_field({_, Len}=Fl, Col, Ll, M, _TInd, _Ind, LD, W)                        end};  pp_field({{field, Name, NameL, F}, _Len}, Col0, Ll, M, TInd, Ind0, LD, W0) ->      {Col, Ind, S, W} = rec_indent(NameL, TInd, Col0, Ind0, W0 + NameL), -    {[Name, " = ", S | pp(F, Col, Ll, M, TInd, Ind, LD, W)], Ll}. % force nl +    Sep = case S of +              [$\n | _] -> " ="; +              _ -> " = " +          end, +    {[Name, Sep, S | pp(F, Col, Ll, M, TInd, Ind, LD, W)], Ll}. % force nl  rec_indent(RInd, TInd, Col0, Ind0, W0) ->      Nl = (TInd > 0) and (RInd > TInd), @@ -291,8 +338,8 @@ write({{list, L}, _}) ->      [$[, write_list(L, $|), $]];  write({{map, Pairs}, _}) ->      [$#,${, write_list(Pairs, $,), $}]; -write({{map_pair, K, V}, _}) -> -    [write(K)," => ",write(V)]; +write({{map_pair, _K, _V}, _}=Pair) -> +    write_pair(Pair);  write({{record, [{Name,_} | L]}, _}) ->      [Name, ${, write_fields(L), $}];  write({{bin, S}, _}) -> @@ -300,6 +347,9 @@ write({{bin, S}, _}) ->  write({S, _}) ->      S. +write_pair({{map_pair, K, V}, _}) -> +    [write(K), " => ", write(V)]. +  write_fields([]) ->      "";  write_fields({dots, _}) -> @@ -333,7 +383,7 @@ write_tail(E, S) ->  %% The depth (D) is used for extracting and counting the characters to  %% print. The structure is kept so that the returned intermediate -%% format can be formatted. The separators (list, tuple, record) are +%% format can be formatted. The separators (list, tuple, record, map) are  %% counted but need to be added later.  %% D =/= 0 @@ -406,21 +456,32 @@ print_length(Term, _D, _RF, _Enc, _Str) ->  print_length_map(_Map, 1, _RF, _Enc, _Str) ->      {"#{...}", 6};  print_length_map(Map, D, RF, Enc, Str) when is_map(Map) -> -    Pairs = print_length_map_pairs(maps:to_list(Map), D, RF, Enc, Str), +    Pairs = print_length_map_pairs(maps_to_list(Map, D), D, RF, Enc, Str),      {{map, Pairs}, list_length(Pairs, 3)}. +maps_to_list(Map, D) when D < 0; map_size(Map) =< D -> +    maps:to_list(Map); +maps_to_list(Map, D) -> +    F = fun(_K, _V, {N, L}) when N =:= D -> +                throw(L); +           (K, V, {N, L}) -> +                {N+1, [{K, V} | L]} +        end, +    lists:reverse(catch maps:fold(F, {0, []}, Map)). +  print_length_map_pairs([], _D, _RF, _Enc, _Str) ->      [];  print_length_map_pairs(_Pairs, 1, _RF, _Enc, _Str) ->      {dots, 3}; -print_length_map_pairs([{K,V}|Pairs], D, RF, Enc, Str) -> -    [print_length_map_pair(K,V,D-1,RF,Enc,Str) | -     print_length_map_pairs(Pairs,D-1,RF,Enc,Str)]. +print_length_map_pairs([{K, V} | Pairs], D, RF, Enc, Str) -> +    [print_length_map_pair(K, V, D - 1, RF, Enc, Str) | +     print_length_map_pairs(Pairs, D - 1, RF, Enc, Str)].  print_length_map_pair(K, V, D, RF, Enc, Str) ->      {KS, KL} = print_length(K, D, RF, Enc, Str),      {VS, VL} = print_length(V, D, RF, Enc, Str), -    {{map_pair, {KS,KL}, {VS,VL}}, KL + VL}. +    KL1 = KL + 4, +    {{map_pair, {KS, KL1}, {VS, VL}}, KL1 + VL}.  print_length_tuple(_Tuple, 1, _RF, _Enc, _Str) ->      {"{...}", 5}; @@ -612,6 +673,8 @@ cind({{tuple,true,L}, _Len}, Col, Ll, M, Ind, LD, W) ->      cind_tag_tuple(L, Col, Ll, M, Ind, LD, W + 1);  cind({{tuple,false,L}, _Len}, Col, Ll, M, Ind, LD, W) ->      cind_list(L, Col + 1, Ll, M, Ind, LD, W + 1); +cind({{map,Pairs},_Len}, Col, Ll, M, Ind, LD, W) -> +    cind_map(Pairs, Col + 2, Ll, M, Ind, LD, W + 2);  cind({{record,[{_Name,NLen} | L]}, _Len}, Col, Ll, M, Ind, LD, W) ->      cind_record(L, NLen, Col, Ll, M, Ind, LD, W + NLen + 1);  cind({{bin,_S}, _Len}, _Col, _Ll, _M, Ind, _LD, _W) -> @@ -637,6 +700,48 @@ cind_tag_tuple([{_Tag,Tlen} | L], Col, Ll, M, Ind, LD, W) ->              throw(no_good)      end. +cind_map([P | Ps], Col, Ll, M, Ind, LD, W) -> +    PW = cind_pair(P, Col, Ll, M, Ind, last_depth(Ps, LD), W), +    cind_pairs_tail(Ps, Col, Col + PW, Ll, M, Ind, LD, W + PW); +cind_map(_, _Col, _Ll, _M, Ind, _LD, _W) -> +    Ind. + +cind_pairs_tail([{_, Len}=P | Ps], Col0, Col, Ll, M, Ind, LD, W) -> +    LD1 = last_depth(Ps, LD), +    ELen = 1 + Len, +    if +        LD1 =:= 0, ELen + 1 < Ll - Col, W + ELen + 1 =< M, ?ATM_PAIR(P); +        LD1 > 0, ELen < Ll - Col - LD1, W + ELen + LD1 =< M, ?ATM_PAIR(P) -> +            cind_pairs_tail(Ps, Col0, Col + ELen, Ll, M, Ind, LD, W + ELen); +        true -> +            PW = cind_pair(P, Col0, Ll, M, Ind, LD1, 0), +            cind_pairs_tail(Ps, Col0, Col0 + PW, Ll, M, Ind, LD, PW) +    end; +cind_pairs_tail(_, _Col0, _Col, _Ll, _M, Ind, _LD, _W) -> +    Ind. + +cind_pair({{map_pair, _Key, _Value}, Len}=Pair, Col, Ll, M, _Ind, LD, W) +         when Len < Ll - Col - LD, Len + W + LD =< M -> +    if +        ?ATM_PAIR(Pair) -> +            Len; +        true -> +            Ll +    end; +cind_pair({{map_pair, K, V}, _Len}, Col0, Ll, M, Ind, LD, W0) -> +    cind(K, Col0, Ll, M, Ind, LD, W0), +    I = map_value_indent(Ind), +    cind(V, Col0 + I, Ll, M, Ind, LD, 0), +    Ll. + +map_value_indent(TInd) -> +    case TInd > 0 of +        true -> +            TInd; +        false -> +            4 +    end. +  cind_record([F | Fs], Nlen, Col0, Ll, M, Ind, LD, W0) ->      Nind = Nlen + 1,      {Col, W} = cind_rec(Nind, Col0, Ll, M, Ind, W0), diff --git a/lib/stdlib/src/qlc.erl b/lib/stdlib/src/qlc.erl index f3665824f2..8c4d835432 100644 --- a/lib/stdlib/src/qlc.erl +++ b/lib/stdlib/src/qlc.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. @@ -1292,6 +1292,10 @@ abstr_term(Fun, Line) when is_function(Fun) ->      end;  abstr_term(PPR, Line) when is_pid(PPR); is_port(PPR); is_reference(PPR) ->      {special, Line, lists:flatten(io_lib:write(PPR))};     +abstr_term(Map, Line) when is_map(Map) -> +    {map,Line, +     [{map_field_assoc,Line,abstr_term(K, Line),abstr_term(V, Line)} || +         {K,V} <- maps:to_list(Map)]};  abstr_term(Simple, Line) ->      erl_parse:abstract(Simple, erl_anno:line(Line)). diff --git a/lib/stdlib/src/zip.erl b/lib/stdlib/src/zip.erl index 340cc21390..fadf96146e 100644 --- a/lib/stdlib/src/zip.erl +++ b/lib/stdlib/src/zip.erl @@ -179,19 +179,6 @@  			 external_attr,  			 local_header_offset}). -%% Unix extra fields (not yet supported) --define(UNIX_EXTRA_FIELD_TAG, 16#000d). --record(unix_extra_field, {atime, -			   mtime, -			   uid, -			   gid}). - -%% extended timestamps (not yet supported) --define(EXTENDED_TIMESTAMP_TAG, 16#5455). -%% -record(extended_timestamp, {mtime, -%% 			     atime, -%% 			     ctime}). -  -define(END_OF_CENTRAL_DIR_MAGIC, 16#06054b50).  -define(END_OF_CENTRAL_DIR_SZ, (4+2+2+2+2+4+4+2)). @@ -381,9 +368,12 @@ do_unzip(F, Options) ->      {Info, In1} = get_central_dir(In0, RawIterator, Input),      %% get rid of zip-comment      Z = zlib:open(), -    Files = get_z_files(Info, Z, In1, Opts, []), -    zlib:close(Z), -    Input(close, In1), +    Files = try +                get_z_files(Info, Z, In1, Opts, []) +            after +                zlib:close(Z), +                Input(close, In1) +            end,      {ok, Files}.  %% Iterate over all files in a zip archive @@ -460,11 +450,20 @@ do_zip(F, Files, Options) ->      #zip_opts{output = Output, open_opts = OpO} = Opts,      Out0 = Output({open, F, OpO}, []),      Z = zlib:open(), -    {Out1, LHS, Pos} = put_z_files(Files, Z, Out0, 0, Opts, []), -    zlib:close(Z), -    Out2 = put_central_dir(LHS, Pos, Out1, Opts), -    Out3 = Output({close, F}, Out2), -    {ok, Out3}. +    try +        {Out1, LHS, Pos} = put_z_files(Files, Z, Out0, 0, Opts, []), +        zlib:close(Z), +        Out2 = put_central_dir(LHS, Pos, Out1, Opts), +        Out3 = Output({close, F}, Out2), +        {ok, Out3} +    catch +        C:R -> +            Stk = erlang:get_stacktrace(), +            zlib:close(Z), +            Output({close, F}, Out0), +            erlang:raise(C, R, Stk) +    end. +  %% List zip directory contents  %% @@ -1379,12 +1378,7 @@ cd_file_header_to_file_info(FileName,  		    gid = 0},      add_extra_info(FI, ExtraField). -%% add extra info to file (some day when we implement it) -add_extra_info(FI, <<?EXTENDED_TIMESTAMP_TAG:16/little, _Rest/binary>>) -> -    FI;     % not yet supported, some other day... -add_extra_info(FI, <<?UNIX_EXTRA_FIELD_TAG:16/little, Rest/binary>>) -> -    _UnixExtra = unix_extra_field_and_var_from_bin(Rest), -    FI;     % not yet supported, and not widely used +%% Currently, we ignore all the extra fields.  add_extra_info(FI, _) ->      FI. @@ -1572,20 +1566,6 @@ dos_date_time_from_datetime({{Year, Month, Day}, {Hour, Min, Sec}}) ->      <<DosDate:16>> = <<YearFrom1980:7, Month:4, Day:5>>,      {DosDate, DosTime}. -unix_extra_field_and_var_from_bin(<<TSize:16/little, -				   ATime:32/little, -				   MTime:32/little, -				   UID:16/little, -				   GID:16/little, -				   Var:TSize/binary>>) -> -    {#unix_extra_field{atime = ATime, -		       mtime = MTime, -		       uid = UID, -		       gid = GID}, -     Var}; -unix_extra_field_and_var_from_bin(_) -> -    throw(bad_unix_extra_field). -  %% A pwrite-like function for iolists (used by memory-option)  pwrite_binary(B, Pos, Bin) when byte_size(B) =:= Pos -> diff --git a/lib/stdlib/test/filename_SUITE.erl b/lib/stdlib/test/filename_SUITE.erl index b7c4d3a6e5..f64ec6acb7 100644 --- a/lib/stdlib/test/filename_SUITE.erl +++ b/lib/stdlib/test/filename_SUITE.erl @@ -29,6 +29,7 @@  	 dirname_bin/1, extension_bin/1, join_bin/1, t_nativename_bin/1]).  -export([pathtype_bin/1,rootname_bin/1,split_bin/1]).  -export([t_basedir_api/1, t_basedir_xdg/1, t_basedir_windows/1]). +-export([safe_relative_path/1]).  -include_lib("common_test/include/ct.hrl"). @@ -41,7 +42,8 @@ all() ->       find_src,       absname_bin, absname_bin_2,       {group,p}, -     t_basedir_xdg, t_basedir_windows]. +     t_basedir_xdg, t_basedir_windows, +     safe_relative_path].  groups() ->       [{p, [parallel], @@ -768,6 +770,71 @@ t_nativename_bin(Config) when is_list(Config) ->                  filename:nativename(<<"/usr/tmp//arne/">>)      end. +safe_relative_path(Config) -> +    PrivDir = proplists:get_value(priv_dir, Config), +    Root = filename:join(PrivDir, ?FUNCTION_NAME), +    ok = file:make_dir(Root), +    ok = file:set_cwd(Root), + +    ok = file:make_dir("a"), +    ok = file:set_cwd("a"), +    ok = file:make_dir("b"), +    ok = file:set_cwd("b"), +    ok = file:make_dir("c"), + +    ok = file:set_cwd(Root), + +    "a" = test_srp("a"), +    "a/b" = test_srp("a/b"), +    "a/b" = test_srp("a/./b"), +    "a/b" = test_srp("a/./b/."), + +    "" = test_srp("a/.."), +    "" = test_srp("a/./.."), +    "" = test_srp("a/../."), +    "a" = test_srp("a/b/.."), +    "a" = test_srp("a/../a"), +    "a" = test_srp("a/../a/../a"), +    "a/b/c" = test_srp("a/../a/b/c"), + +    unsafe = test_srp("a/../.."), +    unsafe = test_srp("a/../../.."), +    unsafe = test_srp("a/./../.."), +    unsafe = test_srp("a/././../../.."), +    unsafe = test_srp("a/b/././../../.."), + +    unsafe = test_srp(PrivDir),                 %Absolute path. + +    ok. + +test_srp(RelPath) -> +    Res = do_test_srp(RelPath), +    Res = case do_test_srp(list_to_binary(RelPath)) of +              Bin when is_binary(Bin) -> +                  binary_to_list(Bin); +              Other -> +                  Other +          end. + +do_test_srp(RelPath) -> +    {ok,Root} = file:get_cwd(), +    ok = file:set_cwd(RelPath), +    {ok,Cwd} = file:get_cwd(), +    ok = file:set_cwd(Root), +    case filename:safe_relative_path(RelPath) of +        unsafe -> +            true = length(Cwd) < length(Root), +            unsafe; +        "" -> +            ""; +        SafeRelPath -> +            ok = file:set_cwd(SafeRelPath), +            {ok,Cwd} = file:get_cwd(), +            true = length(Cwd) >= length(Root), +            ok = file:set_cwd(Root), +            SafeRelPath +    end. +  %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%  %% basedirs  t_basedir_api(Config) when is_list(Config) -> diff --git a/lib/stdlib/test/io_SUITE.erl b/lib/stdlib/test/io_SUITE.erl index 7d48cbc97c..6e99619324 100644 --- a/lib/stdlib/test/io_SUITE.erl +++ b/lib/stdlib/test/io_SUITE.erl @@ -1,7 +1,7 @@  %%  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 1999-2016. All Rights Reserved. +%% Copyright Ericsson AB 1999-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. @@ -30,7 +30,7 @@  	 io_lib_print_binary_depth_one/1, otp_10302/1, otp_10755/1,           otp_10836/1, io_lib_width_too_small/1,           io_with_huge_message_queue/1, format_string/1, -	 maps/1, coverage/1]). +	 maps/1, coverage/1, otp_14175/1]).  -export([pretty/2]). @@ -61,7 +61,7 @@ all() ->       printable_range, bad_printable_range,       io_lib_print_binary_depth_one, otp_10302, otp_10755, otp_10836,       io_lib_width_too_small, io_with_huge_message_queue, -     format_string, maps, coverage]. +     format_string, maps, coverage, otp_14175].  %% Error cases for output.  error_1(Config) when is_list(Config) -> @@ -415,13 +415,13 @@ otp_6354(Config) when is_list(Config) ->      bt(<<"#rrrrr{\n"  	 "    f1 = 1,\n"  	 "    f2 = #rrrrr{f1 = a,f2 = b,f3 = c},\n" -	 "    f3 = \n" +	 "    f3 =\n"  	 "        #rrrrr{\n"  	 "            f1 = h,f2 = i,\n" -	 "            f3 = \n" +	 "            f3 =\n"  	 "                #rrrrr{\n"  	 "                    f1 = aa,\n" -	 "                    f2 = \n" +	 "                    f2 =\n"  	 "                        #rrrrr{\n"  	 "                            f1 = #rrrrr{f1 = a,f2 = b,f3 = c},\n"  	 "                            f2 = 2,f3 = 3},\n" @@ -431,17 +431,17 @@ otp_6354(Config) when is_list(Config) ->  					    2,3},bb}}},  	 -1)),      bt(<<"#d{aaaaaaaaaaaaaaaaaaaa = 1,\n" -	 "   bbbbbbbbbbbbbbbbbbbb = \n" +	 "   bbbbbbbbbbbbbbbbbbbb =\n"  	 "       #d{aaaaaaaaaaaaaaaaaaaa = a,bbbbbbbbbbbbbbbbbbbb = b,\n"  	 "          cccccccccccccccccccc = c,dddddddddddddddddddd = d,\n"  	 "          eeeeeeeeeeeeeeeeeeee = e},\n"  	 "   cccccccccccccccccccc = 3,\n" -	 "   dddddddddddddddddddd = \n" +	 "   dddddddddddddddddddd =\n"  	 "       #d{aaaaaaaaaaaaaaaaaaaa = h,bbbbbbbbbbbbbbbbbbbb = i,\n" -	 "          cccccccccccccccccccc = \n" +	 "          cccccccccccccccccccc =\n"  	 "              #d{aaaaaaaaaaaaaaaaaaaa = aa,"  	 "bbbbbbbbbbbbbbbbbbbb = bb,\n" -	 "                 cccccccccccccccccccc = \n" +	 "                 cccccccccccccccccccc =\n"  	 "                     #d{aaaaaaaaaaaaaaaaaaaa = 1,"  	 "bbbbbbbbbbbbbbbbbbbb = 2,\n"  	 "                        cccccccccccccccccccc = 3," @@ -534,21 +534,21 @@ otp_6354(Config) when is_list(Config) ->         p({A,{A,{A,{A,{A,{A,{A,  			    {g,{h,{i,{j,{k,{l,{m,{n,{o,{a}}}}}}}}}}}}}}}}}, 100)),      bt(<<"#c{\n" -	 " f1 = \n" +	 " f1 =\n"  	 "  #c{\n" -	 "   f1 = \n" +	 "   f1 =\n"  	 "    #c{\n" -	 "     f1 = \n" +	 "     f1 =\n"  	 "      #c{\n" -	 "       f1 = \n" +	 "       f1 =\n"  	 "        #c{\n" -	 "         f1 = \n" +	 "         f1 =\n"  	 "          #c{\n" -	 "           f1 = \n" +	 "           f1 =\n"  	 "            #c{\n" -	 "             f1 = \n" +	 "             f1 =\n"  	 "              #c{\n" -	 "               f1 = \n" +	 "               f1 =\n"  	 "                #c{\n"  	 "                 f1 = #c{f1 = #c{f1 = #c{f1 = a,"  	 "f2 = b},f2 = b},f2 = b},\n" @@ -564,13 +564,13 @@ otp_6354(Config) when is_list(Config) ->         p({c,{c,{c,{c,{c,{c,{c,{c,{c,{c,{c,{c,a,b},b},b},b},b},b},  			 b},b},b},b},b},b}, -1)),      bt(<<"#rrrrr{\n" -	 " f1 = \n" +	 " f1 =\n"  	 "  #rrrrr{\n" -	 "   f1 = \n" +	 "   f1 =\n"  	 "    #rrrrr{\n" -	 "     f1 = \n" +	 "     f1 =\n"  	 "      #rrrrr{\n" -	 "       f1 = \n" +	 "       f1 =\n"  	 "        {rrrrr,{rrrrr,a,#rrrrr{f1 = {rrrrr,1,2},f2 = a,"  	 "f3 = b}},b},\n"  	 "       f2 = {rrrrr,c,d},\n" @@ -2106,3 +2106,200 @@ coverage(_Config) ->      io:format("~s\n", [S2]),      ok. + +otp_14175(_Config) -> +    "..." = p(#{}, 0), +    "#{}" = p(#{}, 1), +    "#{...}" = p(#{a => 1}, 1), +    "#{#{} => a}" = p(#{#{} => a}, 2), +    "#{a => 1,...}" = p(#{a => 1, b => 2}, 2), +    "#{a => 1,b => 2}" = p(#{a => 1, b => 2}, -1), + +    M = #{kaaaaaaaaaaaaaaaaaaa => v1,kbbbbbbbbbbbbbbbbbbb => v2, +          kccccccccccccccccccc => v3,kddddddddddddddddddd => v4, +          keeeeeeeeeeeeeeeeeee => v5}, +    "#{...}" = p(M, 1), +    mt("#{kaaaaaaaaaaaaaaaaaaaa => v1,...}", p(M, 2)), +    mt("#{kaaaaaaaaaaaaaaaaaaaa => 1,kbbbbbbbbbbbbbbbbbbbb => 2,...}", +       p(M, 3)), + +    mt("#{kaaaaaaaaaaaaaaaaaaa => v1,kbbbbbbbbbbbbbbbbbbb => v2,\n" +       "  kccccccccccccccccccc => v3,...}", p(M, 4)), + +    mt("#{kaaaaaaaaaaaaaaaaaaa => v1,kbbbbbbbbbbbbbbbbbbb => v2,\n" +       "  kccccccccccccccccccc => v3,kddddddddddddddddddd => v4,...}", +       p(M, 5)), + +    mt("#{kaaaaaaaaaaaaaaaaaaa => v1,kbbbbbbbbbbbbbbbbbbb => v2,\n" +       "  kccccccccccccccccccc => v3,kddddddddddddddddddd => v4,\n" +       "  keeeeeeeeeeeeeeeeeee => v5}", p(M, 6)), + +    weak("#{aaaaaaaaaaaaaaaaaaa => 1,bbbbbbbbbbbbbbbbbbbb => 2,\n" +         "  cccccccccccccccccccc => {3},\n" +         "  dddddddddddddddddddd => 4,eeeeeeeeeeeeeeeeeeee => 5}", +       p(#{aaaaaaaaaaaaaaaaaaa => 1,bbbbbbbbbbbbbbbbbbbb => 2, +           cccccccccccccccccccc => {3}, +           dddddddddddddddddddd => 4,eeeeeeeeeeeeeeeeeeee => 5}, -1)), + +    M2 = #{dddddddddddddddddddd => {1}, {aaaaaaaaaaaaaaaaaaaa} => 2, +           {bbbbbbbbbbbbbbbbbbbb} => 3,{cccccccccccccccccccc} => 4, +           {eeeeeeeeeeeeeeeeeeee} => 5}, +    "#{...}" = p(M2, 1), +    weak("#{dddddddddddddddddddd => {...},...}", p(M2, 2)), +    weak("#{dddddddddddddddddddd => {1},{...} => 2,...}", p(M2, 3)), + +    weak("#{dddddddddddddddddddd => {1},\n" +         "  {aaaaaaaaaaaaaaaaaaaa} => 2,\n" +         "  {...} => 3,...}", p(M2, 4)), + +    weak("#{dddddddddddddddddddd => {1},\n" +         "  {aaaaaaaaaaaaaaaaaaaa} => 2,\n" +         "  {bbbbbbbbbbbbbbbbbbbb} => 3,\n" +         "  {...} => 4,...}", p(M2, 5)), + +    weak("#{dddddddddddddddddddd => {1},\n" +         "  {aaaaaaaaaaaaaaaaaaaa} => 2,\n" +         "  {bbbbbbbbbbbbbbbbbbbb} => 3,\n" +         "  {cccccccccccccccccccc} => 4,\n" +         "  {...} => 5}", p(M2, 6)), + +    weak("#{dddddddddddddddddddd => {1},\n" +         "  {aaaaaaaaaaaaaaaaaaaa} => 2,\n" +         "  {bbbbbbbbbbbbbbbbbbbb} => 3,\n" +         "  {cccccccccccccccccccc} => 4,\n" +         "  {eeeeeeeeeeeeeeeeeeee} => 5}", p(M2, 7)), + +    M3 = #{kaaaaaaaaaaaaaaaaaaa => vuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuu, +           kbbbbbbbbbbbbbbbbbbb => vvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvv, +           kccccccccccccccccccc => vxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx, +           kddddddddddddddddddd => vyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyy, +           keeeeeeeeeeeeeeeeeee => vzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz}, + +    mt("#{aaaaaaaaaaaaaaaaaaaa =>\n" +       "      uuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuuu,\n" +       "  bbbbbbbbbbbbbbbbbbbb =>\n" +       "      vvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvv,\n" +       "  cccccccccccccccccccc =>\n" +       "      xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx,\n" +       "  dddddddddddddddddddd =>\n" +       "      yyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyy,\n" +       "  eeeeeeeeeeeeeeeeeeee =>\n" +       "      zzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzzz}", p(M3, -1)), + +    R4 = {c,{c,{c,{c,{c,{c,{c,{c,{c,{c,{c,{c,a,b},b},b},b},b},b}, +			 b},b},b},b},b},b}, +    M4 = #{aaaaaaaaaaaaaaaaaaaa => R4, +           bbbbbbbbbbbbbbbbbbbb => R4, +           cccccccccccccccccccc => R4, +           dddddddddddddddddddd => R4, +           eeeeeeeeeeeeeeeeeeee => R4}, + +    weak("#{aaaaaaaaaaaaaaaaaaaa =>\n" +         "      #c{f1 = #c{f1 = #c{...},f2 = b},f2 = b},\n" +         "  bbbbbbbbbbbbbbbbbbbb => #c{f1 = #c{f1 = {...},...},f2 = b},\n" +         "  cccccccccccccccccccc => #c{f1 = #c{...},f2 = b},\n" +         "  dddddddddddddddddddd => #c{f1 = {...},...},\n" +         "  eeeeeeeeeeeeeeeeeeee => #c{...}}", p(M4, 7)), + +    M5 = #{aaaaaaaaaaaaaaaaaaaa => R4}, +    mt("#{aaaaaaaaaaaaaaaaaaaa =>\n" +       "   #c{\n" +       "    f1 =\n" +       "     #c{\n" +       "      f1 =\n" +       "       #c{\n" +       "        f1 =\n" +       "         #c{\n" +       "          f1 =\n" +       "           #c{\n" +       "            f1 =\n" +       "             #c{\n" +       "              f1 =\n" +       "               #c{\n" +       "                f1 =\n" +       "                 #c{\n" +       "                  f1 =\n" +       "                   #c{\n" +       "                    f1 = #c{f1 = #c{f1 = #c{f1 = a,f2 = b},f2 = b}," +                                        "f2 = b},\n" +       "                    f2 = b},\n" +       "                  f2 = b},\n" +       "                f2 = b},\n" +       "              f2 = b},\n" +       "            f2 = b},\n" +       "          f2 = b},\n" +       "        f2 = b},\n" +       "      f2 = b},\n" +       "    f2 = b}}", p(M5, -1)), +    ok. + +%% Just check number of newlines and dots ('...'). +-define(WEAK, true). + +-ifdef(WEAK). + +weak(S, R) -> +    (nl(S) =:= nl(R) andalso +     dots(S) =:= dots(S)). + +nl(S) -> +    [C || C <- S, C =:= $\n]. + +dots(S) -> +    [C || C <- S, C =:= $\.]. + +-else. % WEAK + +weak(S, R) -> +    mt(S, R). + +-endif. % WEAK + +%% If EXACT is defined: mt() matches strings exactly. +%% +%% if EXACT is not defined: do not match the strings exactly, but +%% compare them assuming that all map keys and all map values are +%% equal (by assuming all map keys and all map values have the same +%% length and begin with $k and $v respectively). + +%-define(EXACT, true). + +-ifdef(EXACT). + +mt(S, R) -> +    S =:= R. + +-else. % EXACT + +mt(S, R) -> +    anon(S) =:= anon(R). + +anon(S) -> +    {ok, Ts0, _} = erl_scan:string(S, 1, [text]), +    Ts = anon1(Ts0), +    text(Ts). + +anon1([]) -> []; +anon1([{atom,Anno,Atom}=T|Ts]) -> +    case erl_anno:text(Anno) of +        "k" ++ _ -> +            NewAnno = erl_anno:set_text("key", Anno), +            [{atom,NewAnno,Atom}|anon1(Ts)]; +        "v" ++ _ -> +            NewAnno = erl_anno:set_text("val", Anno), +            [{atom,NewAnno,Atom}|anon1(Ts)]; +        _ -> +            [T|anon1(Ts)] +    end; +anon1([T|Ts]) -> +    [T|anon1(Ts)]. + +text(Ts) -> +    lists:append(text1(Ts)). + +text1([]) -> []; +text1([T|Ts]) -> +    Anno = element(2, T), +    [erl_anno:text(Anno) | text1(Ts)]. + +-endif. % EXACT diff --git a/lib/stdlib/test/qlc_SUITE.erl b/lib/stdlib/test/qlc_SUITE.erl index 2bd940020c..846c2c56f4 100644 --- a/lib/stdlib/test/qlc_SUITE.erl +++ b/lib/stdlib/test/qlc_SUITE.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. @@ -869,11 +869,12 @@ eval_unique(Config) when is_list(Config) ->               [a] = qlc:e(Q2, {unique_all, true})              ">>, -          <<"Q = qlc:q([SQV || SQV <- qlc:q([X || X <- [1,2,1]],unique)], +          <<"Q = qlc:q([SQV || SQV <- qlc:q([X || X <- [1,2,1,#{a => 1}]], +                                             unique)],                         unique),               {call,_,_,[{lc,_,{var,_,'X'},[{generate,_,{var,_,'X'},_}]},_]} =                   qlc:info(Q, [{format,abstract_code},unique_all]), -             [1,2] = qlc:e(Q)">>, +             [1,2,#{a := 1}] = qlc:e(Q)">>,            <<"Q = qlc:q([X || X <- [1,2,1]]),               {call,_,_,[{lc,_,{var,_,'X'},[{generate,_,{var,_,'X'},_}]},_]} = @@ -2620,7 +2621,16 @@ info(Config) when is_list(Config) ->                                        {cons, _, _, _}]},                                {nil,_}}]}]} = i(QH, {format, abstract_code}),            [{5},{6}] = qlc:e(QH), -          [{4},{5},{6}] = qlc:e(F(3))">> +          [{4},{5},{6}] = qlc:e(F(3))">>, + +          <<"Fun = fun ?MODULE:i/2, +             L = [{#{k => #{v => Fun}}, Fun}], +             H = qlc:q([Q || Q <- L, Q =:= {#{k => #{v => Fun}}, Fun}]), +             L = qlc:e(H), +             {call,_,_,[{lc,_,{var,_,'Q'}, +                         [{generate,_,_,_}, +                          {op,_,_,_,_}]}]} = +                qlc:info(H, [{format,abstract_code}])">>         ],      run(Config, Ts), diff --git a/lib/stdlib/test/zip_SUITE.erl b/lib/stdlib/test/zip_SUITE.erl index 7d90795c9e..f0feda217a 100644 --- a/lib/stdlib/test/zip_SUITE.erl +++ b/lib/stdlib/test/zip_SUITE.erl @@ -27,7 +27,7 @@           openzip_api/1, zip_api/1, open_leak/1, unzip_jar/1,  	 unzip_traversal_exploit/1,           compress_control/1, -	 foldl/1]). +	 foldl/1,fd_leak/1]).  -include_lib("common_test/include/ct.hrl").  -include_lib("kernel/include/file.hrl"). @@ -40,7 +40,7 @@ all() ->       unzip_to_binary, zip_to_binary, unzip_options,       zip_options, list_dir_options, aliases, openzip_api,       zip_api, open_leak, unzip_jar, compress_control, foldl, -     unzip_traversal_exploit]. +     unzip_traversal_exploit,fd_leak].  groups() ->       []. @@ -882,3 +882,35 @@ foldl(Config) ->      {error, enoent} = zip:foldl(ZipFun, [], File),      ok. + +fd_leak(Config) -> +    ok = file:set_cwd(proplists:get_value(priv_dir, Config)), +    DataDir = proplists:get_value(data_dir, Config), +    Name = filename:join(DataDir, "bad_file_header.zip"), +    BadExtract = fun() -> +                         {error,bad_file_header} = zip:extract(Name), +                         ok +                 end, +    do_fd_leak(BadExtract, 1), + +    BadCreate = fun() -> +                        {error,enoent} = zip:zip("failed.zip", +                                                 ["none"]), +                        ok +                end, +    do_fd_leak(BadCreate, 1), + +    ok. + +do_fd_leak(_Bad, 10000) -> +    ok; +do_fd_leak(Bad, N) -> +    try Bad() of +        ok -> +            do_fd_leak(Bad, N + 1) +    catch +        C:R -> +            Stk = erlang:get_stacktrace(), +            io:format("Bad error after ~p attempts\n", [N]), +            erlang:raise(C, R, Stk) +    end. diff --git a/lib/xmerl/src/xmerl_sax_parser.erl b/lib/xmerl/src/xmerl_sax_parser.erl index 318a0cf7f4..1aef6c58c4 100644 --- a/lib/xmerl/src/xmerl_sax_parser.erl +++ b/lib/xmerl/src/xmerl_sax_parser.erl @@ -1,7 +1,7 @@  %%--------------------------------------------------------------------  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 2008-2016. All Rights Reserved. +%% Copyright Ericsson AB 2008-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. @@ -33,6 +33,7 @@  %% External exports  %%----------------------------------------------------------------------  -export([file/2, +	 stream/3,  	 stream/2]).  %%---------------------------------------------------------------------- @@ -72,11 +73,12 @@ file(Name,Options) ->              File = filename:basename(Name),  	    ContinuationFun = fun default_continuation_cb/1,              Res = stream(<<>>,  -			[{continuation_fun, ContinuationFun}, -			 {continuation_state, FD},  -			 {current_location, CL}, -			 {entity, File} -			 |Options]), +                         [{continuation_fun, ContinuationFun}, +                          {continuation_state, FD},  +                          {current_location, CL}, +                          {entity, File} +                          |Options], +                         file),  	    ok = file:close(FD),  	    Res      end. @@ -92,19 +94,22 @@ file(Name,Options) ->  %%           EventState = term()  %% Description: Parse a stream containing an XML document.  %%---------------------------------------------------------------------- -stream(Xml, Options) when is_list(Xml), is_list(Options) -> +stream(Xml, Options) -> +    stream(Xml, Options, stream). + +stream(Xml, Options, InputType) when is_list(Xml), is_list(Options) ->      State = parse_options(Options, initial_state()), -    case  State#xmerl_sax_parser_state.file_type of +    case State#xmerl_sax_parser_state.file_type of  	dtd ->  	    xmerl_sax_parser_list:parse_dtd(Xml,   					    State#xmerl_sax_parser_state{encoding = list, -									 input_type = stream}); +									 input_type = InputType});  	normal ->  	    xmerl_sax_parser_list:parse(Xml,   					State#xmerl_sax_parser_state{encoding = list, -								     input_type = stream}) +								     input_type = InputType})      end; -stream(Xml, Options) when is_binary(Xml), is_list(Options) -> +stream(Xml, Options, InputType) when is_binary(Xml), is_list(Options) ->      case parse_options(Options, initial_state()) of   	{error, Reason} -> {error, Reason};  	State -> @@ -127,7 +132,7 @@ stream(Xml, Options) when is_binary(Xml), is_list(Options) ->  				    State#xmerl_sax_parser_state.event_state};  		{Xml1, State1} ->  		    parse_binary(Xml1,  -				 State1#xmerl_sax_parser_state{input_type = stream}, +				 State1#xmerl_sax_parser_state{input_type = InputType},  				 ParseFunction)  	    end      end. @@ -226,12 +231,12 @@ check_encoding_option(E) ->  %% Description: Detects which character set is used in a binary stream.  %%----------------------------------------------------------------------  detect_charset(<<>>, #xmerl_sax_parser_state{continuation_fun = undefined} = _) -> -    throw({error, "Can't detect character encoding due to no indata"}); +    {error, "Can't detect character encoding due to no indata"};  detect_charset(<<>>, #xmerl_sax_parser_state{continuation_fun = CFun,   				      continuation_state = CState} = State) ->      case CFun(CState) of  	{<<>>,  _} -> -	    throw({error, "Can't detect character encoding due to lack of indata"}); +	    {error, "Can't detect character encoding due to lack of indata"};  	{NewBytes, NewContState} ->  	    detect_charset(NewBytes, State#xmerl_sax_parser_state{continuation_state = NewContState})      end; diff --git a/lib/xmerl/src/xmerl_sax_parser.hrl b/lib/xmerl/src/xmerl_sax_parser.hrl index 932ab0cec5..7f9bf6c4d3 100644 --- a/lib/xmerl/src/xmerl_sax_parser.hrl +++ b/lib/xmerl/src/xmerl_sax_parser.hrl @@ -88,14 +88,7 @@  	  current_location,         % Location of the currently parsed XML entity  	  entity,                   % Parsed XML entity  	  skip_external_dtd = false,% If true the external DTD is skipped during parsing -	  input_type                % Source type: file | stream.  -                                    % This field is a preparation for an fix in R17 of a bug in  -                                    % the conformance against the standard. -                                    % Today a file which contains two XML documents will be considered  -                                    % well-formed and the second is placed in the rest part of the  -                                    % return tuple, according to the conformance tests this should fail. -                                    % In the future this will fail if xmerl_sax_aprser:file/2 is used but  -                                    % left to the user in the xmerl_sax_aprser:stream/2 case. +	  input_type                % Source type: file | stream  	 }). diff --git a/lib/xmerl/src/xmerl_sax_parser_base.erlsrc b/lib/xmerl/src/xmerl_sax_parser_base.erlsrc index 4d75805b9b..2b9b37b5f3 100644 --- a/lib/xmerl/src/xmerl_sax_parser_base.erlsrc +++ b/lib/xmerl/src/xmerl_sax_parser_base.erlsrc @@ -1,7 +1,7 @@  %%-*-erlang-*-  %% %CopyrightBegin%  %%  -%% Copyright Ericsson AB 2008-2016. All Rights Reserved. +%% Copyright Ericsson AB 2008-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. @@ -72,7 +72,12 @@ parse(Xml, State) ->  	{ok, Rest, State2} ->  	    State3 =  event_callback(endDocument, State2),  	    ets:delete(RefTable), -	    {ok, State3#xmerl_sax_parser_state.event_state, Rest};  +            case check_if_rest_ok(State3#xmerl_sax_parser_state.input_type, Rest) of +                true -> +                    {ok, State3#xmerl_sax_parser_state.event_state, Rest};  +                false -> +                    format_error(fatal_error, State3, "Input found after legal document") +            end;  	{fatal_error, {State2, Reason}} ->  	    State3 =  event_callback(endDocument, State2),  	    ets:delete(RefTable), @@ -81,10 +86,14 @@ parse(Xml, State) ->  	    State3 =  event_callback(endDocument, State2),  	    ets:delete(RefTable),  	    format_error(Tag, State3, Reason); +	{endDocument, Rest, State2} -> +	    State3 =  event_callback(endDocument, State2), +	    ets:delete(RefTable), +	    {ok, State3#xmerl_sax_parser_state.event_state, Rest};   	Other ->  	    _State2 = event_callback(endDocument, State1),  	    ets:delete(RefTable), -	    throw(Other) +	    {fatal_error, Other}      end.  %%---------------------------------------------------------------------- @@ -111,7 +120,7 @@ parse_dtd(Xml, State) ->  	{Rest, State2} when is_record(State2, xmerl_sax_parser_state) ->  	    State3 =  event_callback(endDocument, State2),  	    ets:delete(RefTable), -	    {ok, State3#xmerl_sax_parser_state.event_state, Rest};  +            {ok, State3#xmerl_sax_parser_state.event_state, Rest};   	{endDocument, Rest, State2} when is_record(State2, xmerl_sax_parser_state) ->  	    State3 =  event_callback(endDocument, State2),  	    ets:delete(RefTable), @@ -119,7 +128,7 @@ parse_dtd(Xml, State) ->  	Other ->  	    _State2 = event_callback(endDocument, State1),  	    ets:delete(RefTable), -	    throw(Other) +	    {fatal_error, Other}      end. @@ -442,6 +451,15 @@ check_if_new_doc_allowed(stream, []) ->  check_if_new_doc_allowed(_, _) ->      false. +check_if_rest_ok(file, []) -> +    true; +check_if_rest_ok(file, <<>>) -> +    true; +check_if_rest_ok(stream, _) -> +    true; +check_if_rest_ok(_, _) -> +    false. +  %%----------------------------------------------------------------------  %% Function: parse_pi_1(Rest, State) -> Result  %% Input:    Rest = string() | binary() @@ -1024,16 +1042,21 @@ parse_etag(Bytes, State) ->      unicode_incomplete_check([Bytes, State, fun parse_etag/2],   			     undefined). -  parse_etag_1(?STRING_REST(">", Rest),   	     #xmerl_sax_parser_state{end_tags=[{_ETag, Uri, LocalName, QName, OldNsList, NewNsList} -					|RestOfETags]} = State, _Tag) -> +					|RestOfETags], +                                    input_type=InputType} = State, _Tag) ->      State1 =  event_callback({endElement, Uri, LocalName, QName}, State),      State2 =  send_end_prefix_mapping_event(NewNsList, State1), -    parse_content(Rest,  -		  State2#xmerl_sax_parser_state{end_tags=RestOfETags, -					 ns = OldNsList}, -		  [], true); +    case check_if_new_doc_allowed(InputType, RestOfETags) of +        true -> +            throw({endDocument, Rest, State2#xmerl_sax_parser_state{ns = OldNsList}}); +        false -> +            parse_content(Rest,  +                          State2#xmerl_sax_parser_state{end_tags=RestOfETags, +                                                        ns = OldNsList}, +                          [], true) +    end;  parse_etag_1(?STRING_UNBOUND_REST(_C, _), State, Tag) ->      {P,TN} = Tag,      ?fatal_error(State, "Bad EndTag: " ++ P ++ ":" ++ TN); @@ -1051,21 +1074,26 @@ parse_etag_1(Bytes, State, Tag) ->  %% Description: Parsing the content part of tags  %%              [43] content ::= (element | CharData | Reference | CDSect | PI | Comment)*  %%---------------------------------------------------------------------- -  parse_content(?STRING_EMPTY, State, Acc, IgnorableWS) -> -    case catch cf(?STRING_EMPTY, State, Acc, IgnorableWS, fun parse_content/4) of -	{Rest, State1} when is_record(State1, xmerl_sax_parser_state) -> -	    {Rest, State1}; -	{fatal_error, {State1, Msg}} -> -	    case check_if_document_complete(State1, Msg) of -		true -> -		    State2 = send_character_event(length(Acc), IgnorableWS, lists:reverse(Acc), State1), -		    {?STRING_EMPTY, State2}; -		false -> -		    ?fatal_error(State1, Msg) -	    end; -	Other -> -	    throw(Other) +    case check_if_document_complete(State, "No more bytes") of +	true -> +	    State1 = send_character_event(length(Acc), IgnorableWS, lists:reverse(Acc), State), +	    {?STRING_EMPTY, State1}; +	false -> +	    case catch cf(?STRING_EMPTY, State, Acc, IgnorableWS, fun parse_content/4) of +		{Rest, State1} when is_record(State1, xmerl_sax_parser_state) -> +		    {Rest, State1}; +		{fatal_error, {State1, Msg}} -> +		    case check_if_document_complete(State1, Msg) of +			true -> +			    State2 = send_character_event(length(Acc), IgnorableWS, lists:reverse(Acc), State1), +			    {?STRING_EMPTY, State2}; +			false -> +			    ?fatal_error(State1, Msg) +		    end; +		Other -> +		    throw(Other) +	    end      end;  parse_content(?STRING("\r") = Bytes, State, Acc, IgnorableWS) ->      cf(Bytes, State, Acc, IgnorableWS, fun parse_content/4); @@ -1094,7 +1122,7 @@ parse_content(?STRING_REST("<?", Rest), State, Acc, IgnorableWS) ->  parse_content(?STRING_REST("<!", Rest1) = Rest, #xmerl_sax_parser_state{end_tags = ET} = State, Acc, IgnorableWS) ->      case ET of   	[] -> -	    {Rest, State}; %%LATH : Skicka ignorable WS ??? +	    {Rest, State}; %% Skicka ignorable WS ???  	_ ->  	    State1 = send_character_event(length(Acc), IgnorableWS, lists:reverse(Acc), State),  	    parse_cdata(Rest1, State1) @@ -1102,7 +1130,7 @@ parse_content(?STRING_REST("<!", Rest1) = Rest, #xmerl_sax_parser_state{end_tags  parse_content(?STRING_REST("<", Rest1) = Rest, #xmerl_sax_parser_state{end_tags = ET} = State, Acc, IgnorableWS) ->      case ET of   	[] -> -	    {Rest, State}; %%LATH :  Skicka ignorable WS ??? +	    {Rest, State}; %% Skicka ignorable WS ???  	_ ->  	    State1 = send_character_event(length(Acc), IgnorableWS, lists:reverse(Acc), State),  	    parse_stag(Rest1, State1) @@ -3290,7 +3318,7 @@ cf(Rest, #xmerl_sax_parser_state{continuation_fun = CFun, continuation_state = C  	catch  	    throw:ErrorTerm ->  		?fatal_error(State, ErrorTerm); -	      exit:Reason -> +            exit:Reason ->  		?fatal_error(State, {'EXIT', Reason})  	end,      case Result of diff --git a/lib/xmerl/test/Makefile b/lib/xmerl/test/Makefile index 7a326e334f..b13fee05b3 100644 --- a/lib/xmerl/test/Makefile +++ b/lib/xmerl/test/Makefile @@ -55,7 +55,8 @@ SUITE_FILES= \  	xmerl_xsd_SUITE.erl \  	xmerl_xsd_MS2002-01-16_SUITE.erl \  	xmerl_xsd_NIST2002-01-16_SUITE.erl \ -	xmerl_xsd_Sun2002-01-16_SUITE.erl +	xmerl_xsd_Sun2002-01-16_SUITE.erl \ +        xmerl_sax_stream_SUITE.erl  XML_FILES= \  	testcases.dtd \ @@ -125,4 +126,5 @@ release_tests_spec: opt  	@tar cfh - xmerl_xsd_MS2002-01-16_SUITE_data | (cd "$(RELSYSDIR)"; tar xf -)  	@tar cfh - xmerl_xsd_NIST2002-01-16_SUITE_data | (cd "$(RELSYSDIR)"; tar xf -)  	@tar cfh - xmerl_xsd_Sun2002-01-16_SUITE_data | (cd "$(RELSYSDIR)"; tar xf -) +	@tar cfh - xmerl_sax_stream_SUITE_data | (cd "$(RELSYSDIR)"; tar xf -)  	chmod -R u+w "$(RELSYSDIR)" diff --git a/lib/xmerl/test/xmerl_sax_SUITE.erl b/lib/xmerl/test/xmerl_sax_SUITE.erl index f5c0a783c4..7d1a70905c 100644 --- a/lib/xmerl/test/xmerl_sax_SUITE.erl +++ b/lib/xmerl/test/xmerl_sax_SUITE.erl @@ -85,17 +85,17 @@ ticket_11551(_Config) ->  <a>hej</a>  <?xml version=\"1.0\" encoding=\"utf-8\" ?>  <a>hej</a>">>, -    {ok, undefined, <<"<?xml",  _/binary>>} = xmerl_sax_parser:stream(Stream1, []), +    {ok, undefined, <<"\n<?xml",  _/binary>>} = xmerl_sax_parser:stream(Stream1, []),      Stream2= <<"<?xml version=\"1.0\" encoding=\"utf-8\" ?>  <a>hej</a>  <?xml version=\"1.0\" encoding=\"utf-8\" ?>  <a>hej</a>">>, -    {ok, undefined, <<"<?xml",  _/binary>>} = xmerl_sax_parser:stream(Stream2, []), +    {ok, undefined, <<"\n\n\n<?xml",  _/binary>>} = xmerl_sax_parser:stream(Stream2, []),      Stream3= <<"<a>hej</a>  <?xml version=\"1.0\" encoding=\"utf-8\" ?>  <a>hej</a>">>, -    {ok, undefined, <<"<?xml",  _/binary>>} = xmerl_sax_parser:stream(Stream3, []), +    {ok, undefined, <<"\n\n<?xml",  _/binary>>} = xmerl_sax_parser:stream(Stream3, []),      ok. diff --git a/lib/xmerl/test/xmerl_sax_std_SUITE.erl b/lib/xmerl/test/xmerl_sax_std_SUITE.erl index 525a3b175a..b8412206cc 100644 --- a/lib/xmerl/test/xmerl_sax_std_SUITE.erl +++ b/lib/xmerl/test/xmerl_sax_std_SUITE.erl @@ -2,7 +2,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. @@ -507,11 +507,8 @@ end_per_testcase(_Func,_Config) ->  'not-wf-sa-036'(Config) ->      file:set_cwd(xmerl_test_lib:get_data_dir(Config)),     Path = filename:join([xmerl_test_lib:get_data_dir(Config),"xmltest","not-wf/sa/036.xml"]), -   %% Special case becase we returns everything after a legal document  -   %% as an rest instead of giving and error to let the user handle  -   %% multipple docs on a stream. -   {ok,_,<<"Illegal data\r\n">>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   %%check_result(R, "not-wf"). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Case  @@ -522,11 +519,8 @@ end_per_testcase(_Func,_Config) ->  'not-wf-sa-037'(Config) ->      file:set_cwd(xmerl_test_lib:get_data_dir(Config)),     Path = filename:join([xmerl_test_lib:get_data_dir(Config),"xmltest","not-wf/sa/037.xml"]), -   %% Special case becase we returns everything after a legal document  -   %% as an rest instead of giving and error to let the user handle  -   %% multipple docs on a stream. -   {ok,_,<<" \r\n">>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   %%check_result(R, "not-wf"). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Case  @@ -561,11 +555,8 @@ end_per_testcase(_Func,_Config) ->  'not-wf-sa-040'(Config) ->      file:set_cwd(xmerl_test_lib:get_data_dir(Config)),     Path = filename:join([xmerl_test_lib:get_data_dir(Config),"xmltest","not-wf/sa/040.xml"]), -   %% Special case becase we returns everything after a legal document  -   %% as an rest instead of giving and error to let the user handle  -   %% multipple docs on a stream. -   {ok,_,<<"<doc></doc>\r\n">>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   %%check_result(R, "not-wf"). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Case  @@ -576,11 +567,8 @@ end_per_testcase(_Func,_Config) ->  'not-wf-sa-041'(Config) ->      file:set_cwd(xmerl_test_lib:get_data_dir(Config)),     Path = filename:join([xmerl_test_lib:get_data_dir(Config),"xmltest","not-wf/sa/041.xml"]), -   %% Special case becase we returns everything after a legal document  -   %% as an rest instead of giving and error to let the user handle  -   %% multipple docs on a stream. -   {ok,_,<<"<doc></doc>\r\n">>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   %%check_result(R, "not-wf"). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Case  @@ -603,11 +591,8 @@ end_per_testcase(_Func,_Config) ->  'not-wf-sa-043'(Config) ->      file:set_cwd(xmerl_test_lib:get_data_dir(Config)),     Path = filename:join([xmerl_test_lib:get_data_dir(Config),"xmltest","not-wf/sa/043.xml"]), -   %% Special case becase we returns everything after a legal document  -   %% as an rest instead of giving and error to let the user handle  -   %% multipple docs on a stream. -   {ok,_,<<"Illegal data\r\n">>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   %%check_result(R, "not-wf"). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Case  @@ -618,11 +603,8 @@ end_per_testcase(_Func,_Config) ->  'not-wf-sa-044'(Config) ->      file:set_cwd(xmerl_test_lib:get_data_dir(Config)),     Path = filename:join([xmerl_test_lib:get_data_dir(Config),"xmltest","not-wf/sa/044.xml"]), -   %% Special case becase we returns everything after a legal document  -   %% as an rest instead of giving and error to let the user handle  -   %% multipple docs on a stream. -   {ok,_,<<"<doc/>\r\n">>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   %%check_result(R, "not-wf"). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Case  @@ -669,11 +651,8 @@ end_per_testcase(_Func,_Config) ->  'not-wf-sa-048'(Config) ->      file:set_cwd(xmerl_test_lib:get_data_dir(Config)),     Path = filename:join([xmerl_test_lib:get_data_dir(Config),"xmltest","not-wf/sa/048.xml"]), -   %% Special case becase we returns everything after a legal document  -   %% as an rest instead of giving and error to let the user handle  -   %% multipple docs on a stream. -   {ok,_,<<"<![CDATA[]]>\r\n">>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   %%check_result(R, "not-wf"). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Case  @@ -1416,11 +1395,8 @@ end_per_testcase(_Func,_Config) ->  'not-wf-sa-110'(Config) ->      file:set_cwd(xmerl_test_lib:get_data_dir(Config)),     Path = filename:join([xmerl_test_lib:get_data_dir(Config),"xmltest","not-wf/sa/110.xml"]), -   %% Special case becase we returns everything after a legal document  -   %% as an rest instead of giving and error to let the user handle  -   %% multipple docs on a stream. -   {ok,_,<<"&e;\r\n">>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   %%check_result(R, "not-wf"). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Case  @@ -1914,9 +1890,9 @@ end_per_testcase(_Func,_Config) ->     %% Special case becase we returns everything after a legal document      %% as an rest instead of giving and error to let the user handle      %% multipple docs on a stream. -   {ok,_,<<"<?xml version=\"1.0\"?>\r\n">>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   % R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), -   % check_result(R, "not-wf"). +   %{ok,_,<<"<?xml version=\"1.0\"?>\r\n">>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Case  @@ -7784,11 +7760,8 @@ end_per_testcase(_Func,_Config) ->  'o-p01fail3'(Config) ->      file:set_cwd(xmerl_test_lib:get_data_dir(Config)),     Path = filename:join([xmerl_test_lib:get_data_dir(Config),"oasis","p01fail3.xml"]), -   %% Special case becase we returns everything after a legal document  -   %% as an rest instead of giving and error to let the user handle  -   %% multipple docs on a stream. -   {ok,_, <<"<bad/>", _/binary>>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   %%check_result(R, "not-wf"). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Case  @@ -11417,12 +11390,8 @@ end_per_testcase(_Func,_Config) ->  'ibm-not-wf-P01-ibm01n02'(Config) ->      file:set_cwd(xmerl_test_lib:get_data_dir(Config)),     Path = filename:join([xmerl_test_lib:get_data_dir(Config),"ibm","not-wf/P01/ibm01n02.xml"]), -   %% Special case becase we returns everything after a legal document  -   %% as an rest instead of giving and error to let the user handle  -   %% multipple docs on a stream. -   {ok,_, <<"<?xml version=\"1.0\"?>", _/binary>>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   % R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), -   % check_result(R, "not-wf"). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Case  @@ -11433,11 +11402,8 @@ end_per_testcase(_Func,_Config) ->  'ibm-not-wf-P01-ibm01n03'(Config) ->      file:set_cwd(xmerl_test_lib:get_data_dir(Config)),     Path = filename:join([xmerl_test_lib:get_data_dir(Config),"ibm","not-wf/P01/ibm01n03.xml"]), -   %% Special case becase we returns everything after a legal document  -   %% as an rest instead of giving and error to let the user handle  -   %% multipple docs on a stream. -   {ok,_, <<"<title>Wrong combination!</title>", _/binary>>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   %%check_result(R, "not-wf"). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Cases  @@ -13027,11 +12993,8 @@ end_per_testcase(_Func,_Config) ->  'ibm-not-wf-P27-ibm27n01'(Config) ->      file:set_cwd(xmerl_test_lib:get_data_dir(Config)),     Path = filename:join([xmerl_test_lib:get_data_dir(Config),"ibm","not-wf/P27/ibm27n01.xml"]), -   %% Special case becase we returns everything after a legal document  -   %% as an rest instead of giving and error to let the user handle  -   %% multipple docs on a stream. -   {ok,_, <<"<!ELEMENT cat EMPTY>">>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   %%check_result(R, "not-wf"). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Cases  @@ -13461,11 +13424,8 @@ end_per_testcase(_Func,_Config) ->  'ibm-not-wf-P39-ibm39n06'(Config) ->      file:set_cwd(xmerl_test_lib:get_data_dir(Config)),     Path = filename:join([xmerl_test_lib:get_data_dir(Config),"ibm","not-wf/P39/ibm39n06.xml"]), -   %% Special case becase we returns everything after a legal document  -   %% as an rest instead of giving and error to let the user handle  -   %% multipple docs on a stream. -   {ok,_,<<"content after end tag\r\n">>} = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]). -   %%check_result(R, "not-wf"). +   R = xmerl_sax_parser:file(Path, [{event_fun, fun(_,_,S) -> S end}]), +   check_result(R, "not-wf").  %%----------------------------------------------------------------------  %% Test Cases  diff --git a/lib/xmerl/test/xmerl_sax_stream_SUITE.erl b/lib/xmerl/test/xmerl_sax_stream_SUITE.erl new file mode 100644 index 0000000000..a306eb66a2 --- /dev/null +++ b/lib/xmerl/test/xmerl_sax_stream_SUITE.erl @@ -0,0 +1,245 @@ +%%-*-erlang-*- +%%---------------------------------------------------------------------- +%% %CopyrightBegin% +%%  +%% Copyright Ericsson AB 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. +%% You may obtain a copy of the License at +%% +%%     http://www.apache.org/licenses/LICENSE-2.0 +%% +%% Unless required by applicable law or agreed to in writing, software +%% distributed under the License is distributed on an "AS IS" BASIS, +%% WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +%% See the License for the specific language governing permissions and +%% limitations under the License. +%%  +%% %CopyrightEnd% +%%---------------------------------------------------------------------- +%% File    : xmerl_sax_stream_SUITE.erl +%%---------------------------------------------------------------------- +-module(xmerl_sax_stream_SUITE). +-compile(export_all). + +%%---------------------------------------------------------------------- +%% Include files +%%---------------------------------------------------------------------- +-include_lib("common_test/include/ct.hrl"). +-include_lib("kernel/include/file.hrl"). + +%%====================================================================== +%% External functions +%%====================================================================== + +%%---------------------------------------------------------------------- +%% Initializations +%%---------------------------------------------------------------------- +all() -> +    [ +     one_document, +     two_documents, +     one_document_and_junk +    ]. + +%%---------------------------------------------------------------------- +%% Initializations +%%---------------------------------------------------------------------- + +init_per_suite(Config) -> +    Config. + +end_per_suite(_Config) -> +    ok. + +init_per_testcase(_TestCase, Config) -> +    Config. + +end_per_testcase(_Func, _Config) -> +    ok. + +%%---------------------------------------------------------------------- +%% Tests +%%---------------------------------------------------------------------- +one_document(Config) -> +    Port = 11111, + +    {ok, ListenSocket} = listen(Port), +    Self = self(), +     +    spawn( +      fun() -> +              case catch gen_tcp:accept(ListenSocket) of +                  {ok, S} -> +                      Result = xmerl_sax_parser:stream(<<>>,  +                                                      [{continuation_state, S}, +                                                       {continuation_fun,  +                                                        fun(Sd) -> +                                                                io:format("Continuation called!!", []), +                                                                case gen_tcp:recv(Sd, 0) of +                                                                    {ok, Packet} -> +                                                                        io:format("Packet: ~p\n", [Packet]), +                                                                        {Packet, Sd}; +                                                                    {error, Reason} -> +                                                                        throw({error, Reason}) +                                                                end +                                                        end}]), +                      Self ! {xmerl_sax, Result}, +                      close(S); +                   Error -> +                      Self ! {xmerl_sax, {error, {accept, Error}}} +              end +      end), + +    {ok, SendSocket} = connect(localhost, Port), +    +    {ok, Binary} = file:read_file(filename:join([datadir(Config), "xmerl_sax_stream_one.xml"])), +     +    send_chunks(SendSocket, Binary), + +    receive +        {xmerl_sax, {ok, undefined, Rest}} -> +            <<"\n">> = Rest, +            io:format("Ok Rest: ~p\n", [Rest])         +       after 5000 -> +               ct:fail("Timeout") +    end, +    ok. +     +two_documents(Config) -> +    Port = 11111, + +    {ok, ListenSocket} = listen(Port), +    Self = self(), +     +    spawn( +      fun() -> +              case catch gen_tcp:accept(ListenSocket) of +                  {ok, S} -> +                      Result = xmerl_sax_parser:stream(<<>>,  +                                                      [{continuation_state, S}, +                                                       {continuation_fun,  +                                                        fun(Sd) -> +                                                                io:format("Continuation called!!", []), +                                                                case gen_tcp:recv(Sd, 0) of +                                                                    {ok, Packet} -> +                                                                        io:format("Packet: ~p\n", [Packet]), +                                                                        {Packet, Sd}; +                                                                    {error, Reason} -> +                                                                        throw({error, Reason}) +                                                                end +                                                        end}]), +                      Self ! {xmerl_sax, Result}, +                      close(S); +                   Error -> +                      Self ! {xmerl_sax, {error, {accept, Error}}} +              end +      end), + +    {ok, SendSocket} = connect(localhost, Port), +    +    {ok, Binary} = file:read_file(filename:join([datadir(Config), "xmerl_sax_stream_two.xml"])), +     +    send_chunks(SendSocket, Binary), + +    receive +        {xmerl_sax, {ok, undefined, Rest}} -> +            <<"\n<?x", _R/binary>> = Rest, +            io:format("Ok Rest: ~p\n", [Rest])         +       after 5000 -> +               ct:fail("Timeout") +    end, +    ok. +     +one_document_and_junk(Config) -> +    Port = 11111, + +    {ok, ListenSocket} = listen(Port), +    Self = self(), +     +    spawn( +      fun() -> +              case catch gen_tcp:accept(ListenSocket) of +                  {ok, S} -> +                      Result = xmerl_sax_parser:stream(<<>>,  +                                                      [{continuation_state, S}, +                                                       {continuation_fun,  +                                                        fun(Sd) -> +                                                                io:format("Continuation called!!", []), +                                                                case gen_tcp:recv(Sd, 0) of +                                                                    {ok, Packet} -> +                                                                        io:format("Packet: ~p\n", [Packet]), +                                                                        {Packet, Sd}; +                                                                    {error, Reason} -> +                                                                        throw({error, Reason}) +                                                                end +                                                        end}]), +                      Self ! {xmerl_sax, Result}, +                      close(S); +                   Error -> +                      Self ! {xmerl_sax, {error, {accept, Error}}} +              end +      end), + +    {ok, SendSocket} = connect(localhost, Port), +    +    {ok, Binary} = file:read_file(filename:join([datadir(Config), "xmerl_sax_stream_one_junk.xml"])), +     +    send_chunks(SendSocket, Binary), + +    receive +        {xmerl_sax, {ok, undefined, Rest}} -> +            <<"\nth", _R/binary>> = Rest, +            io:format("Ok Rest: ~p\n", [Rest])         +       after 10000 -> +               ct:fail("Timeout") +    end, +    ok. +     +%%---------------------------------------------------------------------- +%% Utility functions +%%---------------------------------------------------------------------- +listen(Port) -> +    case catch gen_tcp:listen(Port, [{active, false},  +				     binary,  +				     {keepalive, true}, +				     {reuseaddr,true}]) of +	{ok, ListenSocket} -> +            {ok, ListenSocket}; +	{error, Reason} -> +	    {error, {listen, Reason}} +    end. + +close(Socket) -> +    (catch gen_tcp:close(Socket)). + +connect(Host, Port) -> +     Timeout = 5000, +     % Options1 = check_options(Options), +     Options = [binary], +     case catch gen_tcp:connect(Host, Port, Options, Timeout) of + 	{ok, Socket} -> + 	    {ok, Socket}; + 	{error, Reason} -> +	    {error, Reason} +     end. + +send_chunks(Socket, Binary) -> +    BSize = erlang:size(Binary), +    if  +        BSize > 25 -> +            <<Head:25/binary, Tail/binary>> = Binary, +            case gen_tcp:send(Socket, Head) of +                ok -> +                    timer:sleep(1000), +                    send_chunks(Socket, Tail); +                {error,closed} -> +                    ok +            end; +        true -> +            gen_tcp:send(Socket, Binary) +    end. + +datadir(Config) -> +    proplists:get_value(data_dir, Config). diff --git a/lib/xmerl/test/xmerl_sax_stream_SUITE_data/xmerl_sax_stream_one.xml b/lib/xmerl/test/xmerl_sax_stream_SUITE_data/xmerl_sax_stream_one.xml new file mode 100644 index 0000000000..30328bb188 --- /dev/null +++ b/lib/xmerl/test/xmerl_sax_stream_SUITE_data/xmerl_sax_stream_one.xml @@ -0,0 +1,17 @@ +<?xml version="1.0"?> +<person> +<name> +Arne Andersson +</name> +<address> +<street> + Old Road 456 +</street> +<zip> +12323 +</zip> +<city> +Small City +</city> +</address> +</person> diff --git a/lib/xmerl/test/xmerl_sax_stream_SUITE_data/xmerl_sax_stream_one_junk.xml b/lib/xmerl/test/xmerl_sax_stream_SUITE_data/xmerl_sax_stream_one_junk.xml new file mode 100644 index 0000000000..f730a95865 --- /dev/null +++ b/lib/xmerl/test/xmerl_sax_stream_SUITE_data/xmerl_sax_stream_one_junk.xml @@ -0,0 +1,18 @@ +<?xml version="1.0"?> +<person> +<name> +Arne Andersson +</name> +<address> +<street> + Old Road 456 +</street> +<zip> +12323 +</zip> +<city> +Small City +</city> +</address> +</person> +this is junk ...... diff --git a/lib/xmerl/test/xmerl_sax_stream_SUITE_data/xmerl_sax_stream_two.xml b/lib/xmerl/test/xmerl_sax_stream_SUITE_data/xmerl_sax_stream_two.xml new file mode 100644 index 0000000000..e241a02190 --- /dev/null +++ b/lib/xmerl/test/xmerl_sax_stream_SUITE_data/xmerl_sax_stream_two.xml @@ -0,0 +1,34 @@ +<?xml version="1.0"?> +<person> +<name> +Arne Andersson +</name> +<address> +<street> + Old Road 456 +</street> +<zip> +12323 +</zip> +<city> +Small City +</city> +</address> +</person> +<?xml version="1.0"?> +<person> +<name> +Bertil Bengtson +</name> +<address> +<street> + New Road 4 +</street> +<zip> +12328 +</zip> +<city> +Small City +</city> +</address> +</person> | 
