Non puoi selezionare più di 25 argomenti Gli argomenti devono iniziare con una lettera o un numero, possono includere trattini ('-') e possono essere lunghi fino a 35 caratteri.

233 righe
9.0 KiB

14 anni fa
14 anni fa
14 anni fa
14 anni fa
14 anni fa
14 anni fa
13 anni fa
14 anni fa
13 anni fa
14 anni fa
14 anni fa
13 anni fa
13 anni fa
14 anni fa
14 anni fa
  1. %%% File : ibrowse_test_server.erl
  2. %%% Author : Chandrashekhar Mullaparthi <chandrashekhar.mullaparthi@t-mobile.co.uk>
  3. %%% Description : A server to simulate various test scenarios
  4. %%% Created : 17 Oct 2010 by Chandrashekhar Mullaparthi <chandrashekhar.mullaparthi@t-mobile.co.uk>
  5. -module(ibrowse_test_server).
  6. -export([
  7. start_server/2,
  8. stop_server/1
  9. ]).
  10. -record(request, {method, uri, version, headers = [], body = []}).
  11. -define(dec2hex(X), erlang:integer_to_list(X, 16)).
  12. start_server(Port, Sock_type) ->
  13. Fun = fun() ->
  14. Name = server_proc_name(Port),
  15. register(Name, self()),
  16. case do_listen(Sock_type, Port, [{active, false},
  17. {reuseaddr, true},
  18. {nodelay, true},
  19. {packet, http}]) of
  20. {ok, Sock} ->
  21. do_trace("Server listening on port: ~p~n", [Port]),
  22. accept_loop(Sock, Sock_type);
  23. Err ->
  24. erlang:error(
  25. lists:flatten(
  26. io_lib:format(
  27. "Failed to start server on port ~p. ~p~n",
  28. [Port, Err]))),
  29. exit({listen_error, Err})
  30. end,
  31. unregister(Name)
  32. end,
  33. spawn_link(Fun).
  34. stop_server(Port) ->
  35. server_proc_name(Port) ! stop,
  36. ok.
  37. server_proc_name(Port) ->
  38. list_to_atom("ibrowse_test_server_"++integer_to_list(Port)).
  39. do_listen(tcp, Port, Opts) ->
  40. gen_tcp:listen(Port, Opts);
  41. do_listen(ssl, Port, Opts) ->
  42. application:start(crypto),
  43. application:start(ssl),
  44. ssl:listen(Port, Opts).
  45. do_accept(tcp, Listen_sock) ->
  46. gen_tcp:accept(Listen_sock);
  47. do_accept(ssl, Listen_sock) ->
  48. ssl:ssl_accept(Listen_sock).
  49. accept_loop(Sock, Sock_type) ->
  50. case do_accept(Sock_type, Sock) of
  51. {ok, Conn} ->
  52. Pid = spawn_link(
  53. fun() ->
  54. server_loop(Conn, Sock_type, #request{})
  55. end),
  56. set_controlling_process(Conn, Sock_type, Pid),
  57. Pid ! {setopts, [{active, true}]},
  58. accept_loop(Sock, Sock_type);
  59. Err ->
  60. Err
  61. end.
  62. set_controlling_process(Sock, tcp, Pid) ->
  63. gen_tcp:controlling_process(Sock, Pid);
  64. set_controlling_process(Sock, ssl, Pid) ->
  65. ssl:controlling_process(Sock, Pid).
  66. setopts(Sock, tcp, Opts) ->
  67. inet:setopts(Sock, Opts);
  68. setopts(Sock, ssl, Opts) ->
  69. ssl:setopts(Sock, Opts).
  70. server_loop(Sock, Sock_type, #request{headers = Headers} = Req) ->
  71. receive
  72. {http, Sock, {http_request, HttpMethod, HttpUri, HttpVersion}} ->
  73. server_loop(Sock, Sock_type, Req#request{method = HttpMethod,
  74. uri = HttpUri,
  75. version = HttpVersion});
  76. {http, Sock, {http_header, _, _, _, _} = H} ->
  77. server_loop(Sock, Sock_type, Req#request{headers = [H | Headers]});
  78. {http, Sock, http_eoh} ->
  79. process_request(Sock, Sock_type, Req),
  80. server_loop(Sock, Sock_type, #request{});
  81. {http, Sock, {http_error, Err}} ->
  82. do_trace("Error parsing HTTP request:~n"
  83. "Req so far : ~p~n"
  84. "Err : ", [Req, Err]),
  85. exit({http_error, Err});
  86. {setopts, Opts} ->
  87. setopts(Sock, Sock_type, Opts),
  88. server_loop(Sock, Sock_type, Req);
  89. {tcp_closed, Sock} ->
  90. do_trace("Client closed connection~n", []),
  91. ok;
  92. stop ->
  93. ok;
  94. Other ->
  95. do_trace("Recvd unknown msg: ~p~n", [Other]),
  96. exit({unknown_msg, Other})
  97. after 5000 ->
  98. do_trace("Timing out client connection~n", []),
  99. ok
  100. end.
  101. do_trace(Fmt, Args) ->
  102. do_trace(get(my_trace_flag), Fmt, Args).
  103. do_trace(true, Fmt, Args) ->
  104. io:format("~s -- " ++ Fmt, [ibrowse_lib:printable_date() | Args]);
  105. do_trace(_, _, _) ->
  106. ok.
  107. process_request(Sock, Sock_type,
  108. #request{method='GET',
  109. headers = Headers,
  110. uri = {abs_path, "/ibrowse_stream_once_chunk_pipeline_test"}} = Req) ->
  111. Req_id = case lists:keysearch("X-Ibrowse-Request-Id", 3, Headers) of
  112. false ->
  113. "";
  114. {value, {http_header, _, _, _, Req_id_1}} ->
  115. Req_id_1
  116. end,
  117. Req_id_header = ["x-ibrowse-request-id: ", Req_id, "\r\n"],
  118. do_trace("Recvd req: ~p~n", [Req]),
  119. Body = string:join([integer_to_list(X) || X <- lists:seq(1,100)], "-"),
  120. Chunked_body = chunk_request_body(Body, 50),
  121. Resp_1 = [<<"HTTP/1.1 200 OK\r\n">>,
  122. Req_id_header,
  123. <<"Transfer-Encoding: chunked\r\n\r\n">>],
  124. Resp_2 = Chunked_body,
  125. do_send(Sock, Sock_type, Resp_1),
  126. timer:sleep(100),
  127. do_send(Sock, Sock_type, Resp_2);
  128. process_request(Sock, Sock_type,
  129. #request{method='GET',
  130. headers = _Headers,
  131. uri = {abs_path, "/ibrowse_inac_timeout_test"}} = Req) ->
  132. do_trace("Recvd req: ~p. Sleeping for 30 secs...~n", [Req]),
  133. timer:sleep(30000),
  134. do_trace("...Sending response now.~n", []),
  135. Resp = <<"HTTP/1.1 200 OK\r\nContent-Length: 0\r\n\r\n">>,
  136. do_send(Sock, Sock_type, Resp);
  137. process_request(Sock, Sock_type,
  138. #request{method='HEAD',
  139. headers = _Headers,
  140. uri = {abs_path, "/ibrowse_head_transfer_enc"}}) ->
  141. Resp = <<"HTTP/1.1 400 Bad Request\r\nServer: Apache-Coyote/1.1\r\nContent-Length:5\r\nDate: Wed, 04 Apr 2012 16:53:49 GMT\r\n\r\nabcde">>,
  142. do_send(Sock, Sock_type, Resp);
  143. process_request(Sock, Sock_type,
  144. #request{method='HEAD',
  145. headers = _Headers,
  146. uri = {abs_path, "/ibrowse_head_test"}}) ->
  147. Resp = <<"HTTP/1.1 200 OK\r\nServer: Apache-Coyote/1.1\r\nTransfer-Encoding: chunked\r\nDate: Wed, 04 Apr 2012 16:53:49 GMT\r\nConnection: close\r\n\r\n">>,
  148. do_send(Sock, Sock_type, Resp);
  149. process_request(Sock, Sock_type,
  150. #request{method='POST',
  151. headers = _Headers,
  152. uri = {abs_path, "/ibrowse_303_no_body_test"}}) ->
  153. Resp = <<"HTTP/1.1 303 See Other\r\nLocation: http://example.org\r\n">>,
  154. do_send(Sock, Sock_type, Resp);
  155. process_request(Sock, Sock_type,
  156. #request{method='POST',
  157. headers = _Headers,
  158. uri = {abs_path, "/ibrowse_303_with_body_test"}}) ->
  159. Resp = <<"HTTP/1.1 303 See Other\r\nLocation: http://example.org\r\nContent-Length: 5\r\n\r\nabcde">>,
  160. do_send(Sock, Sock_type, Resp);
  161. process_request(Sock, Sock_type, Req) ->
  162. do_trace("Recvd req: ~p~n", [Req]),
  163. Resp = <<"HTTP/1.1 200 OK\r\nContent-Length: 0\r\n\r\n">>,
  164. do_send(Sock, Sock_type, Resp).
  165. do_send(Sock, tcp, Resp) ->
  166. gen_tcp:send(Sock, Resp);
  167. do_send(Sock, ssl, Resp) ->
  168. ssl:send(Sock, Resp).
  169. %%------------------------------------------------------------------------------
  170. %% Utility functions
  171. %%------------------------------------------------------------------------------
  172. chunk_request_body(Body, _ChunkSize) when is_tuple(Body) orelse
  173. is_function(Body) ->
  174. Body;
  175. chunk_request_body(Body, ChunkSize) ->
  176. chunk_request_body(Body, ChunkSize, []).
  177. chunk_request_body(Body, _ChunkSize, Acc) when Body == <<>>; Body == [] ->
  178. LastChunk = "0\r\n",
  179. lists:reverse(["\r\n", LastChunk | Acc]);
  180. chunk_request_body(Body, ChunkSize, Acc) when is_binary(Body),
  181. size(Body) >= ChunkSize ->
  182. <<ChunkBody:ChunkSize/binary, Rest/binary>> = Body,
  183. Chunk = [?dec2hex(ChunkSize),"\r\n",
  184. ChunkBody, "\r\n"],
  185. chunk_request_body(Rest, ChunkSize, [Chunk | Acc]);
  186. chunk_request_body(Body, _ChunkSize, Acc) when is_binary(Body) ->
  187. BodySize = size(Body),
  188. Chunk = [?dec2hex(BodySize),"\r\n",
  189. Body, "\r\n"],
  190. LastChunk = "0\r\n",
  191. lists:reverse(["\r\n", LastChunk, Chunk | Acc]);
  192. chunk_request_body(Body, ChunkSize, Acc) when length(Body) >= ChunkSize ->
  193. {ChunkBody, Rest} = split_list_at(Body, ChunkSize),
  194. Chunk = [?dec2hex(ChunkSize),"\r\n",
  195. ChunkBody, "\r\n"],
  196. chunk_request_body(Rest, ChunkSize, [Chunk | Acc]);
  197. chunk_request_body(Body, _ChunkSize, Acc) when is_list(Body) ->
  198. BodySize = length(Body),
  199. Chunk = [?dec2hex(BodySize),"\r\n",
  200. Body, "\r\n"],
  201. LastChunk = "0\r\n",
  202. lists:reverse(["\r\n", LastChunk, Chunk | Acc]).
  203. split_list_at(List, N) ->
  204. split_list_at(List, N, []).
  205. split_list_at([], _, Acc) ->
  206. {lists:reverse(Acc), []};
  207. split_list_at(List2, 0, List1) ->
  208. {lists:reverse(List1), List2};
  209. split_list_at([H | List2], N, List1) ->
  210. split_list_at(List2, N-1, [H | List1]).