resp_h.erl 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415
  1. %% This module echoes back the value the test is interested in.
  2. -module(resp_h).
  3. %% @todo Probably should have a separate handler for errors,
  4. %% so that we can dialyze all the other correct calls.
  5. -dialyzer({nowarn_function, do/3}).
  6. -export([init/2]).
  7. init(Req, Opts) ->
  8. do(cowboy_req:binding(key, Req), Req, Opts).
  9. do(<<"set_resp_cookie3">>, Req0, Opts) ->
  10. Req = case cowboy_req:binding(arg, Req0) of
  11. undefined ->
  12. cowboy_req:set_resp_cookie(<<"mycookie">>, "myvalue", Req0);
  13. <<"multiple">> ->
  14. Req1 = cowboy_req:set_resp_cookie(<<"mycookie">>, "myvalue", Req0),
  15. cowboy_req:set_resp_cookie(<<"yourcookie">>, <<"yourvalue">>, Req1);
  16. <<"overwrite">> ->
  17. Req1 = cowboy_req:set_resp_cookie(<<"mycookie">>, "myvalue", Req0),
  18. cowboy_req:set_resp_cookie(<<"mycookie">>, <<"overwrite">>, Req1)
  19. end,
  20. {ok, cowboy_req:reply(200, #{}, "OK", Req), Opts};
  21. do(<<"set_resp_cookie4">>, Req0, Opts) ->
  22. Req = cowboy_req:set_resp_cookie(<<"mycookie">>, "myvalue", Req0,
  23. #{path => cowboy_req:path(Req0)}),
  24. {ok, cowboy_req:reply(200, #{}, "OK", Req), Opts};
  25. do(<<"set_resp_header">>, Req0, Opts) ->
  26. Req = cowboy_req:set_resp_header(<<"content-type">>, <<"text/plain">>, Req0),
  27. {ok, cowboy_req:reply(200, #{}, "OK", Req), Opts};
  28. do(<<"set_resp_header_server">>, Req0, Opts) ->
  29. Req = cowboy_req:set_resp_header(<<"server">>, <<"nginx">>, Req0),
  30. {ok, cowboy_req:reply(200, #{}, "OK", Req), Opts};
  31. do(<<"set_resp_headers">>, Req0, Opts) ->
  32. Req = cowboy_req:set_resp_headers(#{
  33. <<"content-type">> => <<"text/plain">>,
  34. <<"content-encoding">> => <<"compress">>
  35. }, Req0),
  36. {ok, cowboy_req:reply(200, #{}, "OK", Req), Opts};
  37. do(<<"set_resp_headers_http11">>, Req0, Opts) ->
  38. Req = cowboy_req:set_resp_headers(#{
  39. <<"connection">> => <<"custom-header, close">>,
  40. <<"custom-header">> => <<"value">>,
  41. <<"keep-alive">> => <<"timeout=5, max=1000">>,
  42. <<"proxy-connection">> => <<"close">>,
  43. <<"transfer-encoding">> => <<"chunked">>,
  44. <<"upgrade">> => <<"HTTP/1.1">>
  45. }, Req0),
  46. {ok, cowboy_req:reply(200, #{}, "OK", Req), Opts};
  47. do(<<"resp_header_defined">>, Req0, Opts) ->
  48. Req1 = cowboy_req:set_resp_header(<<"content-type">>, <<"text/plain">>, Req0),
  49. <<"text/plain">> = cowboy_req:resp_header(<<"content-type">>, Req1),
  50. <<"text/plain">> = cowboy_req:resp_header(<<"content-type">>, Req1, default),
  51. {ok, cowboy_req:reply(200, #{}, "OK", Req0), Opts};
  52. do(<<"resp_header_default">>, Req, Opts) ->
  53. undefined = cowboy_req:resp_header(<<"content-type">>, Req),
  54. default = cowboy_req:resp_header(<<"content-type">>, Req, default),
  55. {ok, cowboy_req:reply(200, #{}, "OK", Req), Opts};
  56. do(<<"resp_headers">>, Req0, Opts) ->
  57. Req1 = cowboy_req:set_resp_header(<<"server">>, <<"nginx">>, Req0),
  58. Req = cowboy_req:set_resp_headers(#{
  59. <<"content-type">> => <<"text/plain">>,
  60. <<"content-encoding">> => <<"compress">>
  61. }, Req1),
  62. Headers = cowboy_req:resp_headers(Req),
  63. true = maps:is_key(<<"server">>, Headers),
  64. true = maps:is_key(<<"content-type">>, Headers),
  65. true = maps:is_key(<<"content-encoding">>, Headers),
  66. {ok, cowboy_req:reply(200, #{}, "OK", Req), Opts};
  67. do(<<"resp_headers_empty">>, Req, Opts) ->
  68. #{} = cowboy_req:resp_headers(Req),
  69. {ok, cowboy_req:reply(200, #{}, "OK", Req), Opts};
  70. do(<<"set_resp_body">>, Req0, Opts) ->
  71. Arg = cowboy_req:binding(arg, Req0),
  72. Req1 = case Arg of
  73. <<"sendfile0">> ->
  74. AppFile = code:where_is_file("cowboy.app"),
  75. cowboy_req:set_resp_body({sendfile, 0, 0, AppFile}, Req0);
  76. <<"sendfile">> ->
  77. AppFile = code:where_is_file("cowboy.app"),
  78. cowboy_req:set_resp_body({sendfile, 0, filelib:file_size(AppFile), AppFile}, Req0);
  79. _ ->
  80. cowboy_req:set_resp_body(<<"OK">>, Req0)
  81. end,
  82. Req = case Arg of
  83. <<"override">> ->
  84. cowboy_req:reply(200, #{}, <<"OVERRIDE">>, Req1);
  85. _ ->
  86. cowboy_req:reply(200, Req1)
  87. end,
  88. {ok, Req, Opts};
  89. do(<<"has_resp_header">>, Req0, Opts) ->
  90. false = cowboy_req:has_resp_header(<<"content-type">>, Req0),
  91. Req = cowboy_req:set_resp_header(<<"content-type">>, <<"text/plain">>, Req0),
  92. true = cowboy_req:has_resp_header(<<"content-type">>, Req),
  93. {ok, cowboy_req:reply(200, #{}, "OK", Req), Opts};
  94. do(<<"has_resp_body">>, Req0, Opts) ->
  95. case cowboy_req:binding(arg, Req0) of
  96. <<"sendfile">> ->
  97. %% @todo Cases for sendfile. Note that sendfile 0 is unallowed.
  98. false = cowboy_req:has_resp_body(Req0),
  99. Req = cowboy_req:set_resp_body({sendfile, 0, 10, code:where_is_file("cowboy.app")}, Req0),
  100. true = cowboy_req:has_resp_body(Req),
  101. {ok, cowboy_req:reply(200, #{}, <<"OK">>, Req), Opts};
  102. undefined ->
  103. false = cowboy_req:has_resp_body(Req0),
  104. Req = cowboy_req:set_resp_body(<<"OK">>, Req0),
  105. true = cowboy_req:has_resp_body(Req),
  106. {ok, cowboy_req:reply(200, #{}, Req), Opts}
  107. end;
  108. do(<<"delete_resp_header">>, Req0, Opts) ->
  109. %% We try to delete first even though it hasn't been set to
  110. %% make sure this noop is possible.
  111. Req1 = cowboy_req:delete_resp_header(<<"content-type">>, Req0),
  112. false = cowboy_req:has_resp_header(<<"content-type">>, Req1),
  113. Req2 = cowboy_req:set_resp_header(<<"content-type">>, <<"text/plain">>, Req1),
  114. true = cowboy_req:has_resp_header(<<"content-type">>, Req2),
  115. Req = cowboy_req:delete_resp_header(<<"content-type">>, Req2),
  116. false = cowboy_req:has_resp_header(<<"content-type">>, Req),
  117. {ok, cowboy_req:reply(200, #{}, "OK", Req), Opts};
  118. do(<<"inform2">>, Req0, Opts) ->
  119. case cowboy_req:binding(arg, Req0) of
  120. <<"binary">> ->
  121. cowboy_req:inform(<<"102 On my way">>, Req0);
  122. <<"error">> ->
  123. ct_helper:ignore(cowboy_req, inform, 3),
  124. cowboy_req:inform(ok, Req0);
  125. <<"twice">> ->
  126. cowboy_req:inform(102, Req0),
  127. cowboy_req:inform(102, Req0);
  128. Status ->
  129. cowboy_req:inform(binary_to_integer(Status), Req0)
  130. end,
  131. Req = cowboy_req:reply(200, Req0),
  132. {ok, Req, Opts};
  133. do(<<"inform3">>, Req0, Opts) ->
  134. Headers = #{<<"ext-header">> => <<"ext-value">>},
  135. case cowboy_req:binding(arg, Req0) of
  136. <<"binary">> ->
  137. cowboy_req:inform(<<"102 On my way">>, Headers, Req0);
  138. <<"error">> ->
  139. ct_helper:ignore(cowboy_req, inform, 3),
  140. cowboy_req:inform(ok, Headers, Req0);
  141. <<"twice">> ->
  142. cowboy_req:inform(102, Headers, Req0),
  143. cowboy_req:inform(102, Headers, Req0);
  144. Status ->
  145. cowboy_req:inform(binary_to_integer(Status), Headers, Req0)
  146. end,
  147. Req = cowboy_req:reply(200, Req0),
  148. {ok, Req, Opts};
  149. do(<<"reply2">>, Req0, Opts) ->
  150. Req = case cowboy_req:binding(arg, Req0) of
  151. <<"binary">> ->
  152. cowboy_req:reply(<<"200 GOOD">>, Req0);
  153. <<"error">> ->
  154. ct_helper:ignore(cowboy_req, reply, 4),
  155. cowboy_req:reply(ok, Req0);
  156. <<"twice">> ->
  157. ct_helper:ignore(cowboy_req, reply, 4),
  158. Req1 = cowboy_req:reply(200, Req0),
  159. cowboy_req:reply(200, Req1);
  160. Status ->
  161. cowboy_req:reply(binary_to_integer(Status), Req0)
  162. end,
  163. {ok, Req, Opts};
  164. do(<<"reply3">>, Req0, Opts) ->
  165. Req = case cowboy_req:binding(arg, Req0) of
  166. <<"error">> ->
  167. ct_helper:ignore(cowboy_req, reply, 4),
  168. cowboy_req:reply(200, ok, Req0);
  169. Status ->
  170. cowboy_req:reply(binary_to_integer(Status),
  171. #{<<"content-type">> => <<"text/plain">>}, Req0)
  172. end,
  173. {ok, Req, Opts};
  174. do(<<"reply4">>, Req0, Opts) ->
  175. Req = case cowboy_req:binding(arg, Req0) of
  176. <<"error">> ->
  177. ct_helper:ignore(erlang, iolist_size, 1),
  178. cowboy_req:reply(200, #{}, ok, Req0);
  179. <<"204body">> ->
  180. ct_helper:ignore(cowboy_req, reply, 4),
  181. cowboy_req:reply(204, #{}, <<"OK">>, Req0);
  182. <<"304body">> ->
  183. ct_helper:ignore(cowboy_req, reply, 4),
  184. cowboy_req:reply(304, #{}, <<"OK">>, Req0);
  185. Status ->
  186. cowboy_req:reply(binary_to_integer(Status), #{}, <<"OK">>, Req0)
  187. end,
  188. {ok, Req, Opts};
  189. do(<<"stream_reply2">>, Req0, Opts) ->
  190. case cowboy_req:binding(arg, Req0) of
  191. <<"binary">> ->
  192. Req = cowboy_req:stream_reply(<<"200 GOOD">>, Req0),
  193. stream_body(Req),
  194. {ok, Req, Opts};
  195. <<"error">> ->
  196. ct_helper:ignore(cowboy_req, stream_reply, 3),
  197. Req = cowboy_req:stream_reply(ok, Req0),
  198. stream_body(Req),
  199. {ok, Req, Opts};
  200. <<"204">> ->
  201. Req = cowboy_req:stream_reply(204, Req0),
  202. {ok, Req, Opts};
  203. <<"204body">> ->
  204. ct_helper:ignore(cowboy_req, stream_body, 3),
  205. Req = cowboy_req:stream_reply(204, Req0),
  206. stream_body(Req),
  207. {ok, Req, Opts};
  208. <<"304body">> ->
  209. ct_helper:ignore(cowboy_req, stream_body, 3),
  210. Req = cowboy_req:stream_reply(304, Req0),
  211. stream_body(Req),
  212. {ok, Req, Opts};
  213. Status ->
  214. Req = cowboy_req:stream_reply(binary_to_integer(Status), Req0),
  215. stream_body(Req),
  216. {ok, Req, Opts}
  217. end;
  218. do(<<"stream_reply3">>, Req0, Opts) ->
  219. Req = case cowboy_req:binding(arg, Req0) of
  220. <<"error">> ->
  221. ct_helper:ignore(cowboy_req, stream_reply, 3),
  222. cowboy_req:stream_reply(200, ok, Req0);
  223. Status ->
  224. cowboy_req:stream_reply(binary_to_integer(Status),
  225. #{<<"content-type">> => <<"text/plain">>}, Req0)
  226. end,
  227. stream_body(Req),
  228. {ok, Req, Opts};
  229. do(<<"stream_body">>, Req0, Opts) ->
  230. case cowboy_req:binding(arg, Req0) of
  231. <<"fin0">> ->
  232. Req = cowboy_req:stream_reply(200, Req0),
  233. cowboy_req:stream_body(<<"Hello world!">>, nofin, Req),
  234. cowboy_req:stream_body(<<>>, fin, Req),
  235. {ok, Req, Opts};
  236. <<"multiple">> ->
  237. Req = cowboy_req:stream_reply(200, Req0),
  238. cowboy_req:stream_body(<<"Hello ">>, nofin, Req),
  239. cowboy_req:stream_body(<<"world">>, nofin, Req),
  240. cowboy_req:stream_body(<<"!">>, fin, Req),
  241. {ok, Req, Opts};
  242. <<"loop">> ->
  243. Req = cowboy_req:stream_reply(200, Req0),
  244. _ = [cowboy_req:stream_body(<<0:1000000/unit:8>>, nofin, Req)
  245. || _ <- lists:seq(1, 32)],
  246. {ok, Req, Opts};
  247. <<"nofin">> ->
  248. Req = cowboy_req:stream_reply(200, Req0),
  249. cowboy_req:stream_body(<<"Hello world!">>, nofin, Req),
  250. {ok, Req, Opts};
  251. <<"sendfile">> ->
  252. AppFile = code:where_is_file("cowboy.app"),
  253. AppSize = filelib:file_size(AppFile),
  254. Req = cowboy_req:stream_reply(200, Req0),
  255. cowboy_req:stream_body(<<"Hello ">>, nofin, Req),
  256. cowboy_req:stream_body({sendfile, 0, AppSize, AppFile}, nofin, Req),
  257. cowboy_req:stream_body(<<" interspersed ">>, nofin, Req),
  258. cowboy_req:stream_body({sendfile, 0, AppSize, AppFile}, nofin, Req),
  259. cowboy_req:stream_body(<<" world!">>, fin, Req),
  260. {ok, Req, Opts};
  261. <<"sendfile_fin">> ->
  262. AppFile = code:where_is_file("cowboy.app"),
  263. AppSize = filelib:file_size(AppFile),
  264. Req = cowboy_req:stream_reply(200, Req0),
  265. cowboy_req:stream_body(<<"Hello! ">>, nofin, Req),
  266. cowboy_req:stream_body({sendfile, 0, AppSize, AppFile}, fin, Req),
  267. {ok, Req, Opts};
  268. <<"spawn">> ->
  269. Req = cowboy_req:stream_reply(200, Req0),
  270. Parent = self(),
  271. Pid = spawn(fun() ->
  272. cowboy_req:stream_body(<<"Hello ">>, nofin, Req),
  273. cowboy_req:stream_body(<<"world">>, nofin, Req),
  274. cowboy_req:stream_body(<<"!">>, fin, Req),
  275. Parent ! {self(), ok}
  276. end),
  277. receive
  278. {Pid, ok} -> ok
  279. after 5000 ->
  280. error(timeout)
  281. end,
  282. {ok, Req, Opts};
  283. _ ->
  284. %% Call stream_body without initiating streaming.
  285. cowboy_req:stream_body(<<0:800000>>, fin, Req0),
  286. {ok, Req0, Opts}
  287. end;
  288. do(<<"stream_body_content_length">>, Req0, Opts) ->
  289. case cowboy_req:binding(arg, Req0) of
  290. <<"fin0">> ->
  291. Req1 = cowboy_req:set_resp_header(<<"content-length">>, <<"12">>, Req0),
  292. Req = cowboy_req:stream_reply(200, Req1),
  293. cowboy_req:stream_body(<<"Hello world!">>, nofin, Req),
  294. cowboy_req:stream_body(<<>>, fin, Req),
  295. {ok, Req, Opts};
  296. <<"multiple">> ->
  297. Req1 = cowboy_req:set_resp_header(<<"content-length">>, <<"12">>, Req0),
  298. Req = cowboy_req:stream_reply(200, Req1),
  299. cowboy_req:stream_body(<<"Hello ">>, nofin, Req),
  300. cowboy_req:stream_body(<<"world">>, nofin, Req),
  301. cowboy_req:stream_body(<<"!">>, fin, Req),
  302. {ok, Req, Opts};
  303. <<"nofin">> ->
  304. Req1 = cowboy_req:set_resp_header(<<"content-length">>, <<"12">>, Req0),
  305. Req = cowboy_req:stream_reply(200, Req1),
  306. cowboy_req:stream_body(<<"Hello world!">>, nofin, Req),
  307. {ok, Req, Opts};
  308. <<"nofin-error">> ->
  309. Req1 = cowboy_req:set_resp_header(<<"content-length">>, <<"12">>, Req0),
  310. Req = cowboy_req:stream_reply(200, Req1),
  311. cowboy_req:stream_body(<<"Hello">>, nofin, Req),
  312. {ok, Req, Opts}
  313. end;
  314. do(<<"stream_events">>, Req0, Opts) ->
  315. case cowboy_req:binding(arg, Req0) of
  316. %%<<"single">>
  317. %%<<"list">>
  318. <<"single">> ->
  319. Req = cowboy_req:stream_reply(200,
  320. #{<<"content-type">> => <<"text/event-stream">>},
  321. Req0),
  322. cowboy_req:stream_events(#{
  323. event => <<"add_comment">>,
  324. data => <<"Comment text.\nWith many lines.">>
  325. }, fin, Req),
  326. {ok, Req, Opts};
  327. <<"list">> ->
  328. Req = cowboy_req:stream_reply(200,
  329. #{<<"content-type">> => <<"text/event-stream">>},
  330. Req0),
  331. cowboy_req:stream_events([
  332. #{
  333. event => <<"add_comment">>,
  334. data => <<"Comment text.\nWith many lines.">>
  335. },
  336. #{
  337. comment => <<"Set retry higher\nwith many lines also.">>,
  338. retry => 10000
  339. },
  340. #{
  341. id => <<"123">>,
  342. event => <<"add_comment">>,
  343. data => <<"Closing!">>
  344. }
  345. ], fin, Req),
  346. {ok, Req, Opts};
  347. <<"multiple">> ->
  348. Req = cowboy_req:stream_reply(200,
  349. #{<<"content-type">> => <<"text/event-stream">>},
  350. Req0),
  351. cowboy_req:stream_events(#{
  352. event => <<"add_comment">>,
  353. data => <<"Comment text.\nWith many lines.">>
  354. }, nofin, Req),
  355. cowboy_req:stream_events(#{
  356. comment => <<"Set retry higher\nwith many lines also.">>,
  357. retry => 10000
  358. }, nofin, Req),
  359. cowboy_req:stream_events(#{
  360. id => <<"123">>,
  361. event => <<"add_comment">>,
  362. data => <<"Closing!">>
  363. }, fin, Req),
  364. {ok, Req, Opts}
  365. end;
  366. do(<<"stream_trailers">>, Req0, Opts) ->
  367. case cowboy_req:binding(arg, Req0) of
  368. <<"large">> ->
  369. Req = cowboy_req:stream_reply(200, #{
  370. <<"trailer">> => <<"grpc-status">>
  371. }, Req0),
  372. %% The size should be larger than StreamSize and ConnSize
  373. cowboy_req:stream_body(<<0:80000000>>, nofin, Req),
  374. cowboy_req:stream_trailers(#{
  375. <<"grpc-status">> => <<"0">>
  376. }, Req),
  377. {ok, Req, Opts};
  378. _ ->
  379. Req = cowboy_req:stream_reply(200, #{
  380. <<"trailer">> => <<"grpc-status">>
  381. }, Req0),
  382. cowboy_req:stream_body(<<"Hello world!">>, nofin, Req),
  383. cowboy_req:stream_trailers(#{
  384. <<"grpc-status">> => <<"0">>
  385. }, Req),
  386. {ok, Req, Opts}
  387. end;
  388. do(<<"push">>, Req, Opts) ->
  389. case cowboy_req:binding(arg, Req) of
  390. <<"read_body">> ->
  391. cowboy_req:push("/echo/read_body", #{}, Req, #{});
  392. <<"method">> ->
  393. cowboy_req:push("/static/style.css", #{<<"accept">> => <<"text/css">>}, Req,
  394. #{method => <<"HEAD">>});
  395. <<"origin">> ->
  396. cowboy_req:push("/static/style.css", #{<<"accept">> => <<"text/css">>}, Req,
  397. #{scheme => <<"ftp">>, host => <<"127.0.0.1">>, port => 21});
  398. <<"qs">> ->
  399. cowboy_req:push("/static/style.css", #{<<"accept">> => <<"text/css">>}, Req,
  400. #{qs => <<"server=cowboy&version=2.0">>});
  401. _ ->
  402. cowboy_req:push("/static/style.css", #{<<"accept">> => <<"text/css">>}, Req),
  403. %% The text/plain mime is not defined by default, so a 406 will be returned.
  404. cowboy_req:push("/static/plain.txt", #{<<"accept">> => <<"text/plain">>}, Req)
  405. end,
  406. {ok, cowboy_req:reply(200, Req), Opts}.
  407. stream_body(Req) ->
  408. _ = [cowboy_req:stream_body(<<0:800000>>, nofin, Req) || _ <- lists:seq(1,9)],
  409. cowboy_req:stream_body(<<0:800000>>, fin, Req).