123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206 |
- -module(rfc7540_SUITE).
- -compile(export_all).
- -compile(nowarn_export_all).
- -import(ct_helper, [config/2]).
- -import(ct_helper, [doc/1]).
- -import(ct_helper, [get_remote_pid_tcp/1]).
- -import(cowboy_test, [gun_open/1]).
- -import(cowboy_test, [raw_open/1]).
- -import(cowboy_test, [raw_send/2]).
- -import(cowboy_test, [raw_recv_head/1]).
- -import(cowboy_test, [raw_recv/3]).
- all() -> [{group, clear}, {group, tls}].
- groups() ->
- Modules = ct_helper:all(?MODULE),
- Clear = [M || M <- Modules, lists:sublist(atom_to_list(M), 4) =/= "alpn"] -- [prior_knowledge_reject_tls],
- TLS = [M || M <- Modules, lists:sublist(atom_to_list(M), 4) =:= "alpn"] ++ [prior_knowledge_reject_tls],
- [{clear, [parallel], Clear}, {tls, [parallel], TLS}].
- init_per_group(Name = clear, Config) ->
- [{protocol, http2}|cowboy_test:init_http(Name, #{
- env => #{dispatch => cowboy_router:compile(init_routes(Config))},
-
- stream_window_data_threshold => 0
- }, Config)];
- init_per_group(Name = tls, Config) ->
- cowboy_test:init_http2(Name, #{
- env => #{dispatch => cowboy_router:compile(init_routes(Config))},
-
- stream_window_data_threshold => 0
- }, Config).
- end_per_group(Name, _) ->
- ok = cowboy:stop_listener(Name).
- init_routes(_) -> [
- {"localhost", [
- {"/", hello_h, []},
- {"/echo/:key", echo_h, []},
- {"/delay_hello", delay_hello_h, 1200},
- {"/long_polling", long_polling_h, []},
- {"/loop_handler_abort", loop_handler_abort_h, []},
- {"/resp/:key[/:arg]", resp_h, []}
- ]}
- ].
- http_upgrade_ignore_h2(Config) ->
- doc("An h2 token in an Upgrade field must be ignored. (RFC7540 3.2)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- {ok, <<"HTTP/1.1 200">>} = gen_tcp:recv(Socket, 12, 1000),
- ok.
- http_upgrade_ignore_if_http_10(Config) ->
- doc("The Upgrade header must be ignored if part of an HTTP/1.0 request. (RFC7230 6.7)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.0\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- {ok, <<"HTTP/1.1 200">>} = gen_tcp:recv(Socket, 12, 1000),
- ok.
- http_upgrade_ignore_missing_upgrade_in_connection(Config) ->
- doc("The Upgrade header must be listed in the "
- "Connection header field. (RFC7230 6.7)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- {ok, <<"HTTP/1.1 200">>} = gen_tcp:recv(Socket, 12, 1000),
- ok.
- http_upgrade_ignore_missing_http2_settings_in_connection(Config) ->
- doc("The HTTP2-Settings header must be listed in the "
- "Connection header field. (RFC7540 3.2.1, RFC7230 6.7)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- {ok, <<"HTTP/1.1 200">>} = gen_tcp:recv(Socket, 12, 1000),
- ok.
- http_upgrade_ignore_zero_http2_settings_header(Config) ->
- doc("The HTTP Upgrade request must include "
- "exactly one HTTP2-Settings header field (RFC7540 3.2, RFC7540 3.2.1)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "\r\n"]),
- {ok, <<"HTTP/1.1 200">>} = gen_tcp:recv(Socket, 12, 1000),
- ok.
- http_upgrade_reject_two_http2_settings_header(Config) ->
- doc("The HTTP Upgrade request must include "
- "exactly one HTTP2-Settings header field (RFC7540 3.2, RFC7540 3.2.1)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- {ok, <<"HTTP/1.1 400">>} = gen_tcp:recv(Socket, 12, 1000),
- ok.
- http_upgrade_reject_bad_http2_settings_header(Config) ->
- doc("The HTTP Upgrade request must include "
- "a valid HTTP2-Settings header field (RFC7540 3.2, RFC7540 3.2.1)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
-
- "HTTP2-Settings: ", base64:encode(iolist_to_binary(cow_http2:settings(#{}))), "\r\n",
- "\r\n"]),
- {ok, <<"HTTP/1.1 400">>} = gen_tcp:recv(Socket, 12, 1000),
- ok.
- do_recv_101(Socket) ->
- {ok, <<
- "HTTP/1.1 101 Switching Protocols\r\n"
- "connection: Upgrade\r\n"
- "upgrade: h2c\r\n"
- "\r\n"
- >>} = gen_tcp:recv(Socket, 71, 1000),
- ok.
- http_upgrade_101(Config) ->
- doc("A 101 response must be sent on successful upgrade "
- "to HTTP/2 when using the HTTP Upgrade mechanism. (RFC7540 3.2, RFC7230 6.7)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- ok = do_recv_101(Socket),
- ok.
- http_upgrade_server_preface(Config) ->
- doc("The first frame after the upgrade must be a "
- "SETTINGS frame for the server connection preface. (RFC7540 3.2, RFC7540 3.5, RFC7540 6.5)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- ok = do_recv_101(Socket),
-
- {ok, << _:24, 4:8, 0:40 >>} = gen_tcp:recv(Socket, 9, 1000),
- ok.
- http_upgrade_client_preface_timeout(Config) ->
- doc("Clients negotiating HTTP/2 and not sending a preface in "
- "a timely manner must be disconnected."),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- ok = do_recv_101(Socket),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
- {ok, << HeadersSkipLen:24, 1:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, HeadersSkipLen, 1000),
- {ok, << DataSkipLen:24, 0:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, DataSkipLen, 1000),
-
- {error, closed} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- http_upgrade_reject_missing_client_preface(Config) ->
- doc("Servers must treat an invalid connection preface as a "
- "connection error of type PROTOCOL_ERROR. (RFC7540 3.2, RFC7540 3.5)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- ok = do_recv_101(Socket),
-
- ok = gen_tcp:send(Socket, cow_http2:settings(#{})),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
-
-
- Received = lists:reverse(lists:foldl(fun(_, Acc) ->
- case gen_tcp:recv(Socket, 9, 1000) of
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [headers|Acc];
- {ok, << SkipLen:24, 0:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [data|Acc];
- {error, _} ->
- [closed|Acc]
- end
- end, [], [1, 2, 3])),
- case Received of
- [closed|_] -> ok;
- [headers, closed|_] -> ok;
- [headers, data, closed] -> ok
- end.
- http_upgrade_reject_invalid_client_preface(Config) ->
- doc("Servers must treat an invalid connection preface as a "
- "connection error of type PROTOCOL_ERROR. (RFC7540 3.2, RFC7540 3.5)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- ok = do_recv_101(Socket),
-
- ok = gen_tcp:send(Socket, "PRI * HTTP/2.0\r\n\r\nSM: Value\r\n\r\n"),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
-
-
- Received = lists:reverse(lists:foldl(fun(_, Acc) ->
- case gen_tcp:recv(Socket, 9, 1000) of
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [headers|Acc];
- {ok, << SkipLen:24, 0:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [data|Acc];
- {error, _} ->
- [closed|Acc]
- end
- end, [], [1, 2, 3])),
- case Received of
- [closed|_] -> ok;
- [headers, closed|_] -> ok;
- [headers, data, closed] -> ok
- end.
- http_upgrade_reject_missing_client_preface_settings(Config) ->
- doc("Servers must treat an invalid connection preface as a "
- "connection error of type PROTOCOL_ERROR. (RFC7540 3.2, RFC7540 3.5)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- ok = do_recv_101(Socket),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:ping(0)]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
-
-
- Received = lists:reverse(lists:foldl(fun(_, Acc) ->
- case gen_tcp:recv(Socket, 9, 1000) of
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [headers|Acc];
- {ok, << SkipLen:24, 0:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [data|Acc];
- {error, _} ->
- [closed|Acc]
- end
- end, [], [1, 2, 3])),
- case Received of
- [closed|_] -> ok;
- [headers, closed|_] -> ok;
- [headers, data, closed] -> ok
- end.
- http_upgrade_reject_invalid_client_preface_settings(Config) ->
- doc("Servers must treat an invalid connection preface as a "
- "connection error of type PROTOCOL_ERROR. (RFC7540 3.2, RFC7540 3.5)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- ok = do_recv_101(Socket),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", << 0:24, 4:8, 0:9, 1:31 >>]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
-
-
- Received = lists:reverse(lists:foldl(fun(_, Acc) ->
- case gen_tcp:recv(Socket, 9, 1000) of
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [headers|Acc];
- {ok, << SkipLen:24, 0:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [data|Acc];
- {error, _} ->
- [closed|Acc]
- end
- end, [], [1, 2, 3])),
- case Received of
- [closed|_] -> ok;
- [headers, closed|_] -> ok;
- [headers, data, closed] -> ok
- end.
- http_upgrade_accept_client_preface_empty_settings(Config) ->
- doc("The SETTINGS frame in the client preface may be empty. (RFC7540 3.2, RFC7540 3.5)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- ok = do_recv_101(Socket),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
- Received = lists:reverse(lists:foldl(fun(_, Acc) ->
- case gen_tcp:recv(Socket, 9, 1000) of
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [headers|Acc];
- {ok, << SkipLen:24, 0:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [data|Acc];
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} ->
- [settings_ack|Acc]
- end
- end, [], [1, 2, 3])),
- case Received of
- [settings_ack|_] -> ok;
- [headers, settings_ack|_] -> ok;
- [headers, data, settings_ack] -> ok
- end.
- http_upgrade_client_preface_settings_ack_timeout(Config) ->
- doc("The SETTINGS frames sent by the client must be acknowledged. (RFC7540 3.5, RFC7540 6.5.3)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- ok = do_recv_101(Socket),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
-
- Received = lists:reverse(lists:foldl(fun(_, Acc) ->
- case gen_tcp:recv(Socket, 9, 6000) of
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} ->
- Acc;
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [headers|Acc];
- {ok, << SkipLen:24, 0:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [data|Acc];
- {ok, << 8:24, 7:8, 0:40 >>} ->
-
- {ok, << 1:32, 4:32 >>} = gen_tcp:recv(Socket, 8, 1000),
- [goaway|Acc];
- {error, _} ->
-
- Acc
- end
- end, [], [1, 2, 3, 4])),
- case Received of
- [goaway] -> ok;
- [headers, goaway] -> ok;
- [headers, data, goaway] -> ok
- end.
- http_upgrade_response(Config) ->
- doc("A response must be sent to the initial HTTP/1.1 request "
- "after switching to HTTP/2. The response must use "
- "the stream identifier 1. (RFC7540 3.2)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- ok = do_recv_101(Socket),
-
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
- ok = gen_tcp:send(Socket, cow_http2:settings_ack()),
-
- Received = lists:reverse(lists:foldl(fun(_, Acc) ->
- case gen_tcp:recv(Socket, 9, 1000) of
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} ->
- [settings_ack|Acc];
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [headers|Acc];
- {ok, << SkipLen:24, 0:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [data|Acc]
- end
- end, [], [1, 2, 3])),
- case Received of
- [settings_ack, headers, data] -> ok;
- [headers, settings_ack, data] -> ok;
- [headers, data, settings_ack] -> ok
- end.
- http_upgrade_response_half_closed(Config) ->
- doc("The stream for the initial HTTP/1.1 request is half-closed. (RFC7540 3.2)"),
-
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET /long_polling HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- ok = do_recv_101(Socket),
-
- ok = gen_tcp:send(Socket, [
- "PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n",
- cow_http2:settings(#{}),
- cow_http2:data(1, fin, <<"Unexpected DATA frame.">>)
- ]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
-
-
-
- Received = lists:reverse(lists:foldl(fun(_, Acc) ->
- case gen_tcp:recv(Socket, 9, 1000) of
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} ->
- Acc;
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [headers|Acc];
- {ok, << SkipLen:24, 0:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [data|Acc];
- {ok, << 4:24, 3:8, 0:8, 1:32 >>} ->
-
- {ok, << 5:32 >>} = gen_tcp:recv(Socket, 4, 1000),
- [rst_stream|Acc];
- {ok, << 8:24, 7:8, 0:40 >>} ->
-
- {ok, << 1:32, 5:32 >>} = gen_tcp:recv(Socket, 8, 1000),
- [goaway|Acc];
- {error, _} ->
-
- Acc
- end
- end, [], [1, 2, 3, 4])),
- case Received of
- [rst_stream] -> ok;
- [headers, rst_stream] -> ok;
- [headers, data, goaway] -> ok
- end.
- alpn_ignore_h2c(Config) ->
- doc("An h2c ALPN protocol identifier must be ignored. (RFC7540 3.3)"),
- TlsOpts = ct_helper:get_certs_from_ets(),
- {ok, Socket} = ssl:connect("localhost", config(port, Config),
- [{alpn_advertised_protocols, [<<"h2c">>, <<"http/1.1">>]},
- binary, {active, false}|TlsOpts]),
- {ok, <<"http/1.1">>} = ssl:negotiated_protocol(Socket),
- ok.
- alpn_server_preface(Config) ->
- doc("The first frame must be a SETTINGS frame "
- "for the server connection preface. (RFC7540 3.3, RFC7540 3.5, RFC7540 6.5)"),
- TlsOpts = ct_helper:get_certs_from_ets(),
- {ok, Socket} = ssl:connect("localhost", config(port, Config),
- [{alpn_advertised_protocols, [<<"h2">>]},
- binary, {active, false}|TlsOpts]),
- {ok, <<"h2">>} = ssl:negotiated_protocol(Socket),
-
- {ok, << _:24, 4:8, 0:40 >>} = ssl:recv(Socket, 9, 1000),
- ok.
- alpn_client_preface_timeout(Config) ->
- doc("Clients negotiating HTTP/2 and not sending a preface in "
- "a timely manner must be disconnected."),
- TlsOpts = ct_helper:get_certs_from_ets(),
- {ok, Socket} = ssl:connect("localhost", config(port, Config),
- [{alpn_advertised_protocols, [<<"h2">>]},
- binary, {active, false}|TlsOpts]),
- {ok, <<"h2">>} = ssl:negotiated_protocol(Socket),
-
- {ok, << Len:24 >>} = ssl:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = ssl:recv(Socket, 6 + Len, 1000),
-
- {error, closed} = ssl:recv(Socket, 3, 6000),
- ok.
- alpn_reject_missing_client_preface(Config) ->
- doc("Servers must treat an invalid connection preface as a "
- "connection error of type PROTOCOL_ERROR. (RFC7540 3.3, RFC7540 3.5)"),
- TlsOpts = ct_helper:get_certs_from_ets(),
- {ok, Socket} = ssl:connect("localhost", config(port, Config),
- [{alpn_advertised_protocols, [<<"h2">>]},
- binary, {active, false}|TlsOpts]),
- {ok, <<"h2">>} = ssl:negotiated_protocol(Socket),
-
- ok = ssl:send(Socket, cow_http2:settings(#{})),
-
- {ok, << Len:24 >>} = ssl:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = ssl:recv(Socket, 6 + Len, 1000),
-
- {error, closed} = ssl:recv(Socket, 3, 1000),
- ok.
- alpn_reject_invalid_client_preface(Config) ->
- doc("Servers must treat an invalid connection preface as a "
- "connection error of type PROTOCOL_ERROR. (RFC7540 3.3, RFC7540 3.5)"),
- TlsOpts = ct_helper:get_certs_from_ets(),
- {ok, Socket} = ssl:connect("localhost", config(port, Config),
- [{alpn_advertised_protocols, [<<"h2">>]},
- binary, {active, false}|TlsOpts]),
- {ok, <<"h2">>} = ssl:negotiated_protocol(Socket),
-
- ok = ssl:send(Socket, "PRI * HTTP/2.0\r\n\r\nSM: Value\r\n\r\n"),
-
- {ok, << Len:24 >>} = ssl:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = ssl:recv(Socket, 6 + Len, 1000),
-
- {error, closed} = ssl:recv(Socket, 3, 1000),
- ok.
- alpn_reject_missing_client_preface_settings(Config) ->
- doc("Servers must treat an invalid connection preface as a "
- "connection error of type PROTOCOL_ERROR. (RFC7540 3.3, RFC7540 3.5)"),
- TlsOpts = ct_helper:get_certs_from_ets(),
- {ok, Socket} = ssl:connect("localhost", config(port, Config),
- [{alpn_advertised_protocols, [<<"h2">>]},
- binary, {active, false}|TlsOpts]),
- {ok, <<"h2">>} = ssl:negotiated_protocol(Socket),
-
- ok = ssl:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:ping(0)]),
-
- {ok, << Len:24 >>} = ssl:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = ssl:recv(Socket, 6 + Len, 1000),
-
- {error, closed} = ssl:recv(Socket, 3, 1000),
- ok.
- alpn_reject_invalid_client_preface_settings(Config) ->
- doc("Servers must treat an invalid connection preface as a "
- "connection error of type PROTOCOL_ERROR. (RFC7540 3.3, RFC7540 3.5)"),
- TlsOpts = ct_helper:get_certs_from_ets(),
- {ok, Socket} = ssl:connect("localhost", config(port, Config),
- [{alpn_advertised_protocols, [<<"h2">>]},
- binary, {active, false}|TlsOpts]),
- {ok, <<"h2">>} = ssl:negotiated_protocol(Socket),
-
- ok = ssl:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", << 0:24, 4:8, 0:9, 1:31 >>]),
-
- {ok, << Len:24 >>} = ssl:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = ssl:recv(Socket, 6 + Len, 1000),
-
- {error, closed} = ssl:recv(Socket, 3, 1000),
- ok.
- alpn_accept_client_preface_empty_settings(Config) ->
- doc("The SETTINGS frame in the client preface may be empty. (RFC7540 3.3, RFC7540 3.5)"),
- TlsOpts = ct_helper:get_certs_from_ets(),
- {ok, Socket} = ssl:connect("localhost", config(port, Config),
- [{alpn_advertised_protocols, [<<"h2">>]},
- binary, {active, false}|TlsOpts]),
- {ok, <<"h2">>} = ssl:negotiated_protocol(Socket),
-
- ok = ssl:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len:24 >>} = ssl:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = ssl:recv(Socket, 6 + Len, 1000),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = ssl:recv(Socket, 9, 1000),
- ok.
- alpn_client_preface_settings_ack_timeout(Config) ->
- doc("Failure to acknowledge the server's SETTINGS frame "
- "results in a SETTINGS_TIMEOUT connection error. (RFC7540 3.5, RFC7540 6.5.3)"),
- TlsOpts = ct_helper:get_certs_from_ets(),
- {ok, Socket} = ssl:connect("localhost", config(port, Config),
- [{alpn_advertised_protocols, [<<"h2">>]},
- binary, {active, false}|TlsOpts]),
- {ok, <<"h2">>} = ssl:negotiated_protocol(Socket),
-
- ok = ssl:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len:24 >>} = ssl:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = ssl:recv(Socket, 6 + Len, 1000),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = ssl:recv(Socket, 9, 1000),
-
- {ok, << _:24, 7:8, _:72, 4:32 >>} = ssl:recv(Socket, 17, 6000),
- ok.
- alpn(Config) ->
- doc("Successful ALPN negotiation. (RFC7540 3.3)"),
- TlsOpts = ct_helper:get_certs_from_ets(),
- {ok, Socket} = ssl:connect("localhost", config(port, Config),
- [{alpn_advertised_protocols, [<<"h2">>]},
- binary, {active, false}|TlsOpts]),
- {ok, <<"h2">>} = ssl:negotiated_protocol(Socket),
-
-
- ok = ssl:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len:24 >>} = ssl:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = ssl:recv(Socket, 6 + Len, 1000),
-
- ok = ssl:send(Socket, cow_http2:settings_ack()),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = ssl:recv(Socket, 9, 1000),
-
- receive after 6000 -> ok end,
-
- ok = ssl:send(Socket, cow_http2:ping(0)),
-
- {ok, << 8:24, 6:8, 0:7, 1:1, 0:96 >>} = ssl:recv(Socket, 17, 1000),
- ok.
- prior_knowledge_reject_tls(Config) ->
- doc("Implementations that support HTTP/2 over TLS must use ALPN. (RFC7540 3.4)"),
- TlsOpts = ct_helper:get_certs_from_ets(),
- {ok, Socket} = ssl:connect("localhost", config(port, Config),
- [binary, {active, false}|TlsOpts]),
-
- ok = ssl:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
-
- {ok, <<"HTTP/1.1 400">>} = ssl:recv(Socket, 12, 1000),
- ok.
- prior_knowledge_server_preface(Config) ->
- doc("The first frame must be a SETTINGS frame "
- "for the server connection preface. (RFC7540 3.4, RFC7540 3.5, RFC7540 6.5)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << _:24, 4:8, 0:40 >>} = gen_tcp:recv(Socket, 9, 1000),
- ok.
- prior_knowledge_reject_invalid_client_preface(Config) ->
- doc("An incorrect preface is an invalid HTTP/1.1 request. (RFC7540 3.4)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, "PRI * HTTP/2.0\r\n\r\nSM: Value\r\n\r\n"),
-
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
- {error, closed} = gen_tcp:recv(Socket, 9, 1000),
- ok.
- prior_knowledge_reject_missing_client_preface_settings(Config) ->
- doc("Servers must treat an invalid connection preface as a "
- "connection error of type PROTOCOL_ERROR. (RFC7540 3.4, RFC7540 3.5)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:ping(0)]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
- {error, closed} = gen_tcp:recv(Socket, 9, 1000),
- ok.
- prior_knowledge_reject_invalid_client_preface_settings(Config) ->
- doc("Servers must treat an invalid connection preface as a "
- "connection error of type PROTOCOL_ERROR. (RFC7540 3.4, RFC7540 3.5)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", << 0:24, 4:8, 0:9, 1:31 >>]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
- {error, closed} = gen_tcp:recv(Socket, 9, 1000),
- ok.
- prior_knowledge_accept_client_preface_empty_settings(Config) ->
- doc("The SETTINGS frame in the client preface may be empty. (RFC7540 3.4, RFC7540 3.5)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- ok.
- prior_knowledge_client_preface_settings_ack_timeout(Config) ->
- doc("The SETTINGS frames sent by the client must be acknowledged. (RFC7540 3.5, RFC7540 6.5.3)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
- {ok, << _:24, 7:8, _:72, 4:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- do_handshake(Config) ->
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
- ok = gen_tcp:send(Socket, cow_http2:settings_ack()),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, Socket}.
- prior_knowledge(Config) ->
- doc("Streams can be initiated after a successful HTTP/2 connection "
- "with prior knowledge of server capabilities. (RFC7540 3.4)"),
-
- {ok, Socket} = do_handshake(Config),
-
- receive after 6000 -> ok end,
-
- ok = gen_tcp:send(Socket, cow_http2:ping(0)),
-
- {ok, << 8:24, 6:8, 0:7, 1:1, 0:96 >>} = gen_tcp:recv(Socket, 17, 1000),
- ok.
- ignore_unknown_frames(Config) ->
- doc("Frames of unknown type must be ignored and discarded. (RFC7540 4.1)"),
- {ok, Socket} = do_handshake(Config),
-
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- << 10:24, 99:8, 0:40, 0:80 >>,
- cow_http2:data(1, fin, << 0:100/unit:8 >>)
- ]),
-
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {ok, << 100:24, 0:8, 1:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, << 0:100/unit:8 >>} = gen_tcp:recv(Socket, 100, 1000),
- ok.
- ignore_data_unknown_flags(Config) ->
- doc("Undefined DATA frame flags must be ignored. (RFC7540 4.1, RFC7540 6.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- << 100:24, 0:8,
- 1:1, 1:1, 1:1, 1:1,
- 0:1,
- 1:1, 1:1,
- 1:1,
- 0:1, 1:31, 0:100/unit:8 >>
- ]),
-
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {ok, << 100:24, 0:8, 1:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, << 0:100/unit:8 >>} = gen_tcp:recv(Socket, 100, 1000),
- ok.
- ignore_headers_unknown_flags(Config) ->
- doc("Undefined HEADERS frame flags must be ignored. (RFC7540 4.1, RFC7540 6.2)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- Len = iolist_size(HeadersBlock),
- ok = gen_tcp:send(Socket, [
- << Len:24, 1:8,
- 1:1, 1:1,
- 0:1,
- 1:1,
- 0:1,
- 1:1,
- 1:1,
- 0:1,
- 0:1, 1:31 >>,
- HeadersBlock,
- cow_http2:data(1, fin, << 0:100/unit:8 >>)
- ]),
-
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {ok, << 100:24, 0:8, 1:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, << 0:100/unit:8 >>} = gen_tcp:recv(Socket, 100, 1000),
- ok.
- ignore_priority_unknown_flags(Config) ->
- doc("Undefined PRIORITY frame flags must be ignored. (RFC7540 4.1, RFC7540 6.3)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- << 5:24, 2:8,
- 1:1, 1:1, 1:1, 1:1, 1:1, 1:1, 1:1, 1:1,
- 0:1, 1:31, 0:1, 3:31, 0:8 >>,
- cow_http2:data(1, fin, << 0:100/unit:8 >>)
- ]),
-
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {ok, << 100:24, 0:8, 1:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, << 0:100/unit:8 >>} = gen_tcp:recv(Socket, 100, 1000),
- ok.
- ignore_rst_stream_unknown_flags(Config) ->
- doc("Undefined RST_STREAM frame flags must be ignored. (RFC7540 4.1, RFC7540 6.4)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- << 4:24, 3:8,
- 1:1, 1:1, 1:1, 1:1, 1:1, 1:1, 1:1, 1:1,
- 0:1, 1:31, 8:32 >>,
- cow_http2:headers(3, nofin, HeadersBlock),
- cow_http2:data(3, fin, << 0:100/unit:8 >>)
- ]),
-
- {ok, << SkipLen:24, 1:8, _:8, 3:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {ok, << 100:24, 0:8, 1:8, 3:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, << 0:100/unit:8 >>} = gen_tcp:recv(Socket, 100, 1000),
- ok.
- ignore_settings_unknown_flags(Config) ->
- doc("Undefined SETTINGS frame flags must be ignored. (RFC7540 4.1, RFC7540 6.5)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 6:24, 4:8,
- 1:1, 1:1, 1:1, 1:1, 1:1, 1:1, 1:1,
- 0:1,
- 0:32, 2:16, 0:32 >>),
-
- {ok, << 0:24, 4:8, 0:7, 1:1, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- ok.
- ignore_push_promise_unknown_flags(Config) ->
- doc("Undefined PUSH_PROMISE frame flags must be ignored. (RFC7540 4.1, RFC7540 6.6)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 4:24, 5:8,
- 1:1, 1:1, 1:1, 1:1,
- 0:1,
- 1:1,
- 1:1, 1:1,
- 0:1, 1:31, 0:1, 3:31 >>
- ),
-
-
-
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- ignore_ping_unknown_flags(Config) ->
- doc("Undefined PING frame flags must be ignored. (RFC7540 4.1, RFC7540 6.7)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 8:24, 6:8,
- 1:1, 1:1, 1:1, 1:1, 1:1, 1:1, 1:1,
- 0:1,
- 0:32, 0:64 >>),
-
- {ok, << 8:24, 6:8, _:7, 1:1, _:32, 0:64 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- ignore_goaway_unknown_flags(Config) ->
- doc("Undefined GOAWAY frame flags must be ignored. (RFC7540 4.1, RFC7540 6.8)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 8:24, 7:8,
- 1:1, 1:1, 1:1, 1:1, 1:1, 1:1, 1:1, 1:1,
- 0:32, 0:64 >>),
-
- {ok, << _:24, 7:8, _:72, 0:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- ignore_window_update_unknown_flags(Config) ->
- doc("Undefined WINDOW_UPDATE frame flags must be ignored. (RFC7540 4.1, RFC7540 6.9)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 4:24, 8:8,
- 1:1, 1:1, 1:1, 1:1, 1:1, 1:1, 1:1, 1:1,
- 0:32, 1000:32 >>),
-
- ok = gen_tcp:send(Socket, cow_http2:ping(0)),
-
- {ok, << 8:24, 6:8, _:7, 1:1, _:32, 0:64 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- ignore_continuation_unknown_flags(Config) ->
- doc("Undefined CONTINUATION frame flags must be ignored. (RFC7540 4.1, RFC7540 6.10)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- Len = iolist_size(HeadersBlock),
- ok = gen_tcp:send(Socket, [
- << 0:24, 1:8, 0:8, 0:1, 1:31 >>,
- << Len:24, 9:8,
- 1:1, 1:1, 1:1, 1:1, 1:1,
- 1:1,
- 1:1, 1:1,
- 0:1, 1:31 >>,
- HeadersBlock,
- cow_http2:data(1, fin, << 0:100/unit:8 >>)
- ]),
-
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {ok, << 100:24, 0:8, 1:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, << 0:100/unit:8 >>} = gen_tcp:recv(Socket, 100, 1000),
- ok.
- ignore_data_reserved_bit(Config) ->
- doc("Reserved 1-bit field of DATA frame must be ignored. (RFC7540 4.1, RFC7540 6.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- << 100:24, 0:8, 0:7, 1:1,
- 1:1,
- 1:31, 0:100/unit:8 >>
- ]),
-
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {ok, << 100:24, 0:8, 1:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, << 0:100/unit:8 >>} = gen_tcp:recv(Socket, 100, 1000),
- ok.
- ignore_headers_reserved_bit(Config) ->
- doc("Reserved 1-bit field of HEADERS frame must be ignored. (RFC7540 4.1, RFC7540 6.2)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- Len = iolist_size(HeadersBlock),
- ok = gen_tcp:send(Socket, [
- << Len:24, 1:8, 0:5, 1:1, 0:2,
- 1:1,
- 1:31 >>,
- HeadersBlock,
- cow_http2:data(1, fin, << 0:100/unit:8 >>)
- ]),
-
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {ok, << 100:24, 0:8, 1:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, << 0:100/unit:8 >>} = gen_tcp:recv(Socket, 100, 1000),
- ok.
- ignore_priority_reserved_bit(Config) ->
- doc("Reserved 1-bit field of PRIORITY frame must be ignored. (RFC7540 4.1, RFC7540 6.3)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- << 5:24, 2:8, 0:8,
- 1:1,
- 1:31, 0:1, 3:31, 0:8 >>,
- cow_http2:data(1, fin, << 0:100/unit:8 >>)
- ]),
-
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {ok, << 100:24, 0:8, 1:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, << 0:100/unit:8 >>} = gen_tcp:recv(Socket, 100, 1000),
- ok.
- ignore_rst_stream_reserved_bit(Config) ->
- doc("Reserved 1-bit field of RST_STREAM frame must be ignored. (RFC7540 4.1, RFC7540 6.4)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- << 4:24, 3:8, 0:8,
- 1:1,
- 1:31, 8:32 >>,
- cow_http2:headers(3, nofin, HeadersBlock),
- cow_http2:data(3, fin, << 0:100/unit:8 >>)
- ]),
-
- {ok, << SkipLen:24, 1:8, _:8, 3:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {ok, << 100:24, 0:8, 1:8, 3:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, << 0:100/unit:8 >>} = gen_tcp:recv(Socket, 100, 1000),
- ok.
- ignore_settings_reserved_bit(Config) ->
- doc("Reserved 1-bit field of SETTINGS frame must be ignored. (RFC7540 4.1, RFC7540 6.5)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 6:24, 4:8, 0:8,
- 1:1,
- 0:31, 2:16, 0:32 >>),
-
- {ok, << 0:24, 4:8, 0:7, 1:1, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- ok.
- ignore_push_promise_reserved_bit(Config) ->
- doc("Reserved 1-bit field of PUSH_PROMISE frame must be ignored. (RFC7540 4.1, RFC7540 6.6)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 4:24, 5:8, 0:5, 1:1, 0:2,
- 1:1,
- 1:31, 0:1, 3:31 >>
- ),
-
-
-
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- ignore_ping_reserved_bit(Config) ->
- doc("Reserved 1-bit field of PING frame must be ignored. (RFC7540 4.1, RFC7540 6.7)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 8:24, 6:8, 0:8,
- 1:1,
- 0:31, 0:64 >>),
-
- {ok, << 8:24, 6:8, _:7, 1:1, _:32, 0:64 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- ignore_goaway_reserved_bit(Config) ->
- doc("Reserved 1-bit field of GOAWAY frame must be ignored. (RFC7540 4.1, RFC7540 6.8)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 8:24, 7:8, 0:8,
- 1:1,
- 0:31, 0:64 >>),
-
- {ok, << _:24, 7:8, _:72, 0:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- ignore_window_update_reserved_bit(Config) ->
- doc("Reserved 1-bit field of WINDOW_UPDATE frame must be ignored. (RFC7540 4.1, RFC7540 6.9)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 4:24, 8:8, 0:8,
- 1:1,
- 0:31, 1000:32 >>),
-
- ok = gen_tcp:send(Socket, cow_http2:ping(0)),
-
- {ok, << 8:24, 6:8, _:7, 1:1, _:32, 0:64 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- ignore_continuation_reserved_bit(Config) ->
- doc("Reserved 1-bit field of CONTINUATION frame must be ignored. (RFC7540 4.1, RFC7540 6.10)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- Len = iolist_size(HeadersBlock),
- ok = gen_tcp:send(Socket, [
- << 0:24, 1:8, 0:8, 0:1, 1:31 >>,
- << Len:24, 9:8, 0:5, 1:1, 0:2,
- 1:1,
- 1:31 >>,
- HeadersBlock,
- cow_http2:data(1, fin, << 0:100/unit:8 >>)
- ]),
-
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {ok, << 100:24, 0:8, 1:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, << 0:100/unit:8 >>} = gen_tcp:recv(Socket, 100, 1000),
- ok.
- max_frame_size_allow_exactly_default(Config) ->
- doc("All implementations must allow frame sizes of at least 16384. (RFC7540 4.1, RFC7540 4.2)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, fin, << 0:16384/unit:8 >>)
- ]),
-
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} = case gen_tcp:recv(Socket, 9, 1000) of
-
- {ok, <<4:24, 8:8, 0:40>>} ->
- {ok, _} = gen_tcp:recv(Socket, 4 + 13, 1000),
- gen_tcp:recv(Socket, 9, 1000);
- Res ->
- Res
- end,
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {ok, << 16384:24, 0:8, 1:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, << 0:16384/unit:8 >>} = gen_tcp:recv(Socket, 16384, 1000),
- ok.
- max_frame_size_reject_larger_than_default(Config) ->
- doc("A FRAME_SIZE_ERROR connection error must be sent when receiving "
- "frames larger than the default 16384 length. (RFC7540 4.1, RFC7540 4.2)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, fin, << 0:16385/unit:8 >>)
- ]),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- max_frame_size_allow_exactly_custom(Config0) ->
- doc("An endpoint that sets SETTINGS_MAX_FRAME_SIZE must allow frames "
- "of up to that size. (RFC7540 4.2, RFC7540 6.5.2)"),
-
- Config = cowboy_test:init_http(?FUNCTION_NAME, #{
- env => #{dispatch => cowboy_router:compile(init_routes(Config0))},
- max_frame_size_received => 30000
- }, Config0),
- try
-
- {ok, Socket} = do_handshake(Config),
-
-
- Headers = [
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ],
- {HeadersBlock, _} = cow_hpack:encode(Headers),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, fin, <<0:30000/unit:8>>)
- ]),
-
- {ok, << Len2:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Len2, 6000),
-
- {error, timeout} = gen_tcp:recv(Socket, 0, 1000)
- after
- cowboy:stop_listener(?FUNCTION_NAME)
- end.
- max_frame_size_reject_larger_than_custom(Config0) ->
- doc("An endpoint that sets SETTINGS_MAX_FRAME_SIZE must reject frames "
- "of up to that size with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.5.2)"),
-
- Config = cowboy_test:init_http(?FUNCTION_NAME, #{
- env => #{dispatch => cowboy_router:compile(init_routes(Config0))},
- max_frame_size_received => 30000
- }, Config0),
- try
-
- {ok, Socket} = do_handshake(Config),
-
-
- Headers = [
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ],
- {HeadersBlock, _} = cow_hpack:encode(Headers),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, fin, <<0:30001/unit:8>>)
- ]),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000)
- after
- cowboy:stop_listener(?FUNCTION_NAME)
- end.
- data_reject_frame_size_0_padded_flag(Config) ->
- doc("DATA frames of size 0 with the PADDED flag set must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- << 0:24, 0:8, 0:4, 1:1, 0:2, 1:1, 0:1, 1:31 >>
- ]),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- data_reject_frame_size_too_small_padded_flag(Config) ->
- doc("DATA frames with Pad Length >= Length must be rejected "
- "with a PROTOCOL_ERROR connection error. (RFC7540 6.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- << 10:24, 0:8, 0:4, 1:1, 0:2, 1:1, 0:1, 1:31, 10:8, 0:80 >>
- ]),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- headers_reject_frame_size_0_padded_flag(Config) ->
- doc("HEADERS frames of size 0 with the PADDED flag set must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.2)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 0:24, 1:8, 0:4, 1:1, 0:2, 1:1, 0:1, 1:31 >>),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- headers_reject_frame_size_too_small_padded_flag(Config) ->
- doc("HEADERS frames with no priority flag and Pad Length >= Length "
- "must be rejected with a PROTOCOL_ERROR connection error. (RFC7540 6.2)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 10:24, 1:8, 0:4, 1:1, 0:2, 1:1, 0:1, 1:31, 10:8, 0:80 >>),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- headers_reject_frame_size_too_small_priority_flag(Config) ->
- doc("HEADERS frames of size smaller than 5 with the PRIORITY flag set must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.2)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 4:24, 1:8,
- 0:2, 1:1, 0:4, 1:1, 0:1, 1:31, 0:1, 3:31, 0:8 >>),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- headers_reject_frame_size_5_padded_and_priority_flags(Config) ->
- doc("HEADERS frames of size smaller than 6 with the PADDED "
- "and PRIORITY flags set must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.2)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 5:24, 1:8,
- 0:2, 1:1, 0:1, 1:1, 0:2, 1:1, 0:1, 1:31, 0:8, 0:1, 3:31, 0:8 >>),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- headers_reject_frame_size_too_small_padded_and_priority_flags(Config) ->
- doc("HEADERS frames of size smaller than Length+6 with the PADDED and PRIORITY flags set "
- "must be rejected with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.2)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 15:24, 1:8,
- 0:2, 1:1, 0:1, 1:1, 0:2, 1:1, 0:1, 1:31, 10:8, 0:1, 3:31, 0:8, 0:80 >>),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- priority_reject_frame_size_too_small(Config) ->
- doc("PRIORITY frames of size smaller than 5 must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.3)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 4:24, 2:8, 0:9, 1:31, 0:1, 3:31, 0:8 >>),
-
- {ok, << _:24, 3:8, _:40, 6:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- priority_reject_frame_size_too_large(Config) ->
- doc("PRIORITY frames of size larger than 5 must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.3)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 6:24, 2:8, 0:9, 1:31, 0:1, 3:31, 0:16 >>),
-
- {ok, << _:24, 3:8, _:40, 6:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- rst_stream_reject_frame_size_too_small(Config) ->
- doc("RST_STREAM frames of size smaller than 4 must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.4)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, fin, HeadersBlock),
- << 3:24, 3:8, 0:9, 1:31, 8:32 >>
- ]),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- rst_stream_reject_frame_size_too_large(Config) ->
- doc("RST_STREAM frames of size larger than 4 must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.4)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, fin, HeadersBlock),
- << 5:24, 3:8, 0:9, 1:31, 8:32 >>
- ]),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- settings_reject_bad_frame_size(Config) ->
- doc("SETTINGS frames must have a size multiple of 6 or be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.5)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 5:24, 4:8, 0:40, 1:16, 4096:32 >>),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- settings_ack_reject_non_empty_frame_size(Config) ->
- doc("SETTINGS frames with the ACK flag set and a non-empty payload "
- "must be rejected with a FRAME_SIZE_ERROR connection error (RFC7540 4.2, RFC7540 6.5)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
- ok = gen_tcp:send(Socket, << 6:24, 4:8, 0:7, 1:1, 0:32, 1:16, 4096:32 >>),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- push_promise_reject_frame_size_too_small(Config) ->
- doc("PUSH_PROMISE frames of size smaller than 4 must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.6)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 3:24, 5:8, 0:5, 1:1, 0:3, 1:31, 0:1, 3:31 >>),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- push_promise_reject_frame_size_4_padded_flag(Config) ->
- doc("PUSH_PROMISE frames of size smaller than 5 with the PADDED flag set must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.6)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 4:24, 5:8, 0:4, 1:1, 1:1, 0:3, 1:31, 0:1, 0:8, 3:31 >>),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- push_promise_reject_frame_size_too_small_padded_flag(Config) ->
- doc("PUSH_PROMISE frames of size smaller than Length+5 with the PADDED flag set "
- "must be rejected with a PROTOCOL_ERROR connection error. (RFC7540 4.2, RFC7540 6.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- Len = 14 + iolist_size(HeadersBlock),
- ok = gen_tcp:send(Socket, [
- << Len:24, 5:8, 0:4, 1:1, 1:1, 0:3, 1:31, 10:8, 0:1, 3:31 >>,
- HeadersBlock,
- << 0:80 >>
- ]),
-
-
-
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- ping_reject_frame_size_too_small(Config) ->
- doc("PING frames of size smaller than 8 must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.7)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 7:24, 6:8, 0:40, 0:56 >>),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- ping_reject_frame_size_too_large(Config) ->
- doc("PING frames of size larger than 8 must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.7)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 9:24, 6:8, 0:40, 0:72 >>),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- goaway_reject_frame_size_too_small(Config) ->
- doc("GOAWAY frames of size smaller than 8 must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.8)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 7:24, 7:8, 0:40, 0:56 >>),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- goaway_allow_frame_size_too_large(Config) ->
- doc("GOAWAY frames of size larger than 8 must be allowed. (RFC7540 6.8)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 12:24, 7:8, 0:40, 0:64, 99999:32 >>),
-
- {ok, << _:24, 7:8, _:72, 0:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- window_update_reject_frame_size_too_small(Config) ->
- doc("WINDOW_UPDATE frames of size smaller than 4 must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.9)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 3:24, 8:8, 0:40, 1000:24 >>),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- window_update_reject_frame_size_too_large(Config) ->
- doc("WINDOW_UPDATE frames of size larger than 4 must be rejected "
- "with a FRAME_SIZE_ERROR connection error. (RFC7540 4.2, RFC7540 6.9)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 5:24, 8:8, 0:40, 1000:40 >>),
-
- {ok, << _:24, 7:8, _:72, 6:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- headers_compression_error(Config) ->
- doc("A decoding error in a HEADERS frame's header block must be rejected "
- "with a COMPRESSION_ERROR connection error. (RFC7540 4.3, RFC7540 6.2)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 10:24, 1:8, 0:5, 1:1, 0:1, 1:1, 0:1, 1:31, 0:10/unit:8 >>),
-
- {ok, << _:24, 7:8, _:72, 9:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- continuation_compression_error(Config) ->
- doc("A decoding error in a CONTINUATION frame's header block must be rejected "
- "with a COMPRESSION_ERROR connection error. (RFC7540 4.3, RFC7540 6.10)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, [
- << 0:24, 1:8, 0:7, 1:1, 0:1, 1:31 >>,
- << 10:24, 9:8, 0:5, 1:1, 0:3, 1:31, 0:10/unit:8 >>
- ]),
-
- {ok, << _:24, 7:8, _:72, 9:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- continuation_with_frame_interleaved_error(Config) ->
- doc("Frames interleaved in a header block must be rejected "
- "with a PROTOCOL_ERROR connection error. (RFC7540 4.3, RFC7540 6.2, RFC7540 6.10)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, [
- << 0:24, 1:8, 0:7, 1:1, 0:1, 1:31 >>,
- cow_http2:ping(0)
- ]),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- continuation_wrong_stream_error(Config) ->
- doc("CONTINUATION frames with an incorrect stream identifier must be rejected "
- "with a PROTOCOL_ERROR connection error. (RFC7540 4.3, RFC7540 6.2)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, [
- << 0:24, 1:8, 0:7, 1:1, 0:1, 1:31 >>,
- << 0:24, 9:8, 0:9, 3:31 >>
- ]),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- idle_stream_reject_data(Config) ->
- doc("DATA frames received on an idle stream must be rejected "
- "with a PROTOCOL_ERROR connection error. (RFC7540 5.1, RFC7540 6.1)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, cow_http2:data(1, fin, <<"Unexpected DATA frame.">>)),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- idle_stream_accept_headers(Config) ->
- doc("HEADERS frames received on an idle stream must be accepted. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- idle_stream_accept_priority(Config) ->
- doc("PRIORITY frames received on an idle stream must be accepted. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, cow_http2:priority(1, shared, 3, 123)),
-
- {error, timeout} = gen_tcp:recv(Socket, 7, 1000),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- idle_stream_reject_rst_stream(Config) ->
- doc("RST_STREAM frames received on an idle stream must be rejected "
- "with a PROTOCOL_ERROR connection error. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, cow_http2:rst_stream(1, no_error)),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- idle_stream_reject_push_promise(Config) ->
- doc("PUSH_PROMISE frames received on an idle stream must be rejected "
- "with a PROTOCOL_ERROR connection error. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:push_promise(1, 3, HeadersBlock)),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- idle_stream_reject_window_update(Config) ->
- doc("WINDOW_UPDATE frames received on an idle stream must be rejected "
- "with a PROTOCOL_ERROR connection error. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, cow_http2:window_update(1, 12345)),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- half_closed_remote_reject_data(Config) ->
- doc("DATA frames received on a half-closed (remote) stream must be rejected "
- "with a STREAM_CLOSED stream error. (RFC7540 5.1, RFC7540 6.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- ok = gen_tcp:send(Socket, cow_http2:data(1, fin, <<"Unexpected DATA frame.">>)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 5:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- half_closed_remote_reject_headers(Config) ->
- doc("HEADERS frames received on a half-closed (remote) stream must be rejected "
- "with a STREAM_CLOSED connection error. (RFC7540 4.3, RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 7:8, _:72, 5:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- half_closed_remote_accept_priority(Config) ->
- doc("PRIORITY frames received on a half-closed stream must be accepted. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- ok = gen_tcp:send(Socket, cow_http2:priority(1, shared, 3, 123)),
-
- {ok, << _:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- half_closed_remote_accept_rst_stream(Config) ->
- doc("RST_STREAM frames received on a half-closed stream must be accepted. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- ok = gen_tcp:send(Socket, cow_http2:rst_stream(1, no_error)),
-
- {error, timeout} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- half_closed_remote_accept_window_update(Config) ->
- doc("WINDOW_UPDATE frames received on a half-closed stream must be accepted. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- ok = gen_tcp:send(Socket, cow_http2:window_update(1, 12345)),
-
- {ok, << _:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- rst_stream_closed_reject_data(Config) ->
- doc("DATA frames received on a stream closed via RST_STREAM must be rejected "
- "with a STREAM_CLOSED connection error. (RFC7540 5.1, RFC7540 6.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, nofin, HeadersBlock)),
-
- ok = gen_tcp:send(Socket, cow_http2:rst_stream(1, cancel)),
-
- ok = gen_tcp:send(Socket, cow_http2:data(1, fin, <<"Unexpected DATA frame.">>)),
-
- {ok, << _:24, 7:8, _:72, 5:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- rst_stream_closed_reject_headers(Config) ->
- doc("HEADERS frames received on a stream closed via RST_STREAM must be rejected "
- "with a STREAM_CLOSED connection error. (RFC7540 4.3, RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, nofin, HeadersBlock)),
-
- ok = gen_tcp:send(Socket, cow_http2:rst_stream(1, cancel)),
-
- ok = gen_tcp:send(Socket, cow_http2:headers(1, nofin, HeadersBlock)),
-
- {ok, << _:24, 7:8, _:72, 5:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- rst_stream_closed_accept_priority(Config) ->
- doc("PRIORITY frames received on a stream closed via RST_STREAM "
- "must be accepted. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, nofin, HeadersBlock)),
-
- ok = gen_tcp:send(Socket, cow_http2:rst_stream(1, cancel)),
-
- ok = gen_tcp:send(Socket, cow_http2:priority(1, shared, 3, 123)),
-
- {error, timeout} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- rst_stream_closed_ignore_rst_stream(Config) ->
- doc("RST_STREAM frames received on a stream closed via RST_STREAM "
- "must be ignored to avoid looping. (RFC7540 5.1, RFC7540 5.4.2)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, nofin, HeadersBlock)),
-
- ok = gen_tcp:send(Socket, cow_http2:rst_stream(1, cancel)),
-
- ok = gen_tcp:send(Socket, cow_http2:rst_stream(1, cancel)),
-
- {error, timeout} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- rst_stream_closed_reject_window_update(Config) ->
- doc("WINDOW_UPDATE frames received on a stream closed via RST_STREAM "
- "must be rejected with a STREAM_CLOSED stream error. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, nofin, HeadersBlock)),
-
- ok = gen_tcp:send(Socket, cow_http2:rst_stream(1, cancel)),
-
- ok = gen_tcp:send(Socket, cow_http2:window_update(1, 12345)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 5:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- stream_closed_reject_data(Config) ->
- doc("DATA frames received on a stream closed normally must be rejected "
- "with a STREAM_CLOSED connection error. (RFC7540 5.1, RFC7540 6.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << Length1:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Length1, 6000),
- {ok, << Length2:24, 0:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Length2, 6000),
-
- ok = gen_tcp:send(Socket, cow_http2:data(1, fin, <<"Unexpected DATA frame.">>)),
-
- {ok, << _:24, 7:8, _:72, 5:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- stream_closed_reject_headers(Config) ->
- doc("HEADERS frames received on a stream closed normally must be rejected "
- "with a STREAM_CLOSED connection error. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << Length1:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Length1, 6000),
- {ok, << Length2:24, 0:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Length2, 6000),
-
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 7:8, _:72, 5:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- stream_closed_accept_priority(Config) ->
- doc("PRIORITY frames received on a stream closed normally must be accepted. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << Length1:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Length1, 6000),
- {ok, << Length2:24, 0:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Length2, 6000),
-
- ok = gen_tcp:send(Socket, cow_http2:priority(1, shared, 3, 123)),
-
- {error, timeout} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- stream_closed_accept_rst_stream(Config) ->
- doc("RST_STREAM frames received on a stream closed normally "
- "must be accepted for a short period. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << Length1:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Length1, 6000),
- {ok, << Length2:24, 0:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Length2, 6000),
-
- ok = gen_tcp:send(Socket, cow_http2:rst_stream(1, cancel)),
-
- {error, timeout} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- stream_closed_accept_window_update(Config) ->
- doc("WINDOW_UPDATE frames received on a stream closed normally "
- "must be accepted for a short period. (RFC7540 5.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << Length1:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Length1, 6000),
- {ok, << Length2:24, 0:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Length2, 6000),
-
- ok = gen_tcp:send(Socket, cow_http2:window_update(1, 12345)),
-
- {error, timeout} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- reject_streamid_even(Config) ->
- doc("HEADERS frames received with an even-numbered streamid "
- "must be rejected with a PROTOCOL_ERROR connection error. (RFC7540 5.1.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(2, fin, HeadersBlock)),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- reject_streamid_0(Config) ->
- doc("HEADERS frames received with streamid 0 (zero) "
- "must be rejected with a PROTOCOL_ERROR connection error. (RFC7540 5.1.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(0, fin, HeadersBlock)),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- http_upgrade_reject_reuse_streamid_1(Config) ->
- doc("Attempts to reuse streamid 1 after upgrading to HTTP/2 "
- "must be rejected with a STREAM_CLOSED connection error. (RFC7540 5.1.1)"),
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
- ok = gen_tcp:send(Socket, [
- "GET / HTTP/1.1\r\n"
- "Host: localhost\r\n"
- "Connection: Upgrade, HTTP2-Settings\r\n"
- "Upgrade: h2c\r\n"
- "HTTP2-Settings: ", base64:encode(cow_http2:settings_payload(#{})), "\r\n",
- "\r\n"]),
- ok = do_recv_101(Socket),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
- ok = gen_tcp:send(Socket, cow_http2:settings_ack()),
-
- Received = lists:reverse(lists:foldl(fun(_, Acc) ->
- case gen_tcp:recv(Socket, 9, 1000) of
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} ->
- [settings_ack|Acc];
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [headers|Acc];
- {ok, << SkipLen:24, 0:8, _:8, 1:32 >>} ->
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- [data|Acc]
- end
- end, [], [1, 2, 3])),
- case Received of
- [settings_ack, headers, data] -> ok;
- [headers, settings_ack, data] -> ok;
- [headers, data, settings_ack] -> ok
- end,
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 7:8, _:72, 5:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- reject_streamid_lower(Config) ->
- doc("HEADERS frames received with streamid lower than the previous stream "
- "must be rejected with a STREAM_CLOSED connection error. (RFC7540 5.1.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(5, fin, HeadersBlock)),
-
- {ok, << Length1:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Length1, 6000),
- {ok, << Length2:24, 0:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Length2, 6000),
-
- ok = gen_tcp:send(Socket, cow_http2:headers(3, fin, HeadersBlock)),
-
- {ok, << _:24, 7:8, _:72, 5:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- reject_self_dependent_stream_headers(Config) ->
- doc("HEADERS frames opening a stream that depends on itself "
- "must be rejected with a PROTOCOL_ERROR connection error. (RFC7540 5.3.1)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 5:24, 1:8,
- 0:2, 1:1, 0:4, 1:1, 0:1, 1:31, 0:1, 1:31, 0:8 >>),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- reject_self_dependent_stream_headers_with_padding(Config) ->
- doc("HEADERS frames opening a stream that depends on itself "
- "must be rejected with a PROTOCOL_ERROR connection error. (RFC7540 5.3.1)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 6:24, 1:8,
- 0:2, 1:1, 0:1, 1:1, 0:2, 1:1, 0:1, 1:31, 0:8, 0:1, 1:31, 0:8 >>),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- reject_self_dependent_stream_priority(Config) ->
- doc("PRIORITY frames making a stream depend on itself "
- "must be rejected with a PROTOCOL_ERROR stream error. (RFC7540 5.3.1)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, cow_http2:priority(1, shared, 1, 123)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- continuation_with_extension_frame_interleaved_error(Config) ->
- doc("Extension frames interleaved in a header block must be rejected "
- "with a PROTOCOL_ERROR connection error. "
- "(RFC7540 4.3, RFC7540 5.5, RFC7540 6.2, RFC7540 6.10)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, [
- << 0:24, 1:8, 0:7, 1:1, 0:1, 1:31 >>,
- << 0:24, 128:8, 0:8, 0:32 >>
- ]),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- settings_header_table_size_client(Config) ->
- doc("The SETTINGS_HEADER_TABLE_SIZE setting can be used to "
- "inform the server of the maximum header table size "
- "used by the client to decode header blocks. (RFC7540 6.5.2)"),
- HeaderTableSize = 128,
-
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n",
- cow_http2:settings(#{header_table_size => HeaderTableSize})]),
-
- {ok, << Len0:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len0/binary >>} = gen_tcp:recv(Socket, 6 + Len0, 1000),
-
- ok = gen_tcp:send(Socket, cow_http2:settings_ack()),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
- DecodeState = cow_hpack:set_max_size(HeaderTableSize, cow_hpack:init()),
- EncodeState = cow_hpack:init(),
-
- {ReqHeadersBlock1, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ], EncodeState),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, ReqHeadersBlock1)),
-
- {ok, << Len1:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, RespHeadersBlock1} = gen_tcp:recv(Socket, Len1, 6000),
- {RespHeaders, _} = cow_hpack:decode(RespHeadersBlock1, DecodeState),
- {_, <<"200">>} = lists:keyfind(<<":status">>, 1, RespHeaders),
-
-
- ok.
- settings_header_table_size_server(Config0) ->
- doc("The SETTINGS_HEADER_TABLE_SIZE setting can be used to "
- "inform the client of the maximum header table size "
- "used by the server to decode header blocks. (RFC7540 6.5.2)"),
- HeaderTableSize = 128,
-
- Config = cowboy_test:init_http(?FUNCTION_NAME, #{
- env => #{dispatch => cowboy_router:compile(init_routes(Config0))},
- max_decode_table_size => HeaderTableSize
- }, Config0),
- try
-
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n",
- cow_http2:settings(#{header_table_size => HeaderTableSize})]),
-
- {ok, << Len0:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, Data = <<_:48, _:Len0/binary>>} = gen_tcp:recv(Socket, 6 + Len0, 1000),
-
- {ok, {settings, #{header_table_size := HeaderTableSize}}, <<>>}
- = cow_http2:parse(<<Len0:24, Data/binary>>),
-
- ok = gen_tcp:send(Socket, cow_http2:settings_ack()),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
- DecodeState = cow_hpack:init(),
- EncodeState = cow_hpack:set_max_size(HeaderTableSize, cow_hpack:init()),
-
- {ReqHeadersBlock1, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ], EncodeState),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, ReqHeadersBlock1)),
-
- {ok, << Len1:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, RespHeadersBlock1} = gen_tcp:recv(Socket, Len1, 6000),
- {RespHeaders, _} = cow_hpack:decode(RespHeadersBlock1, DecodeState),
- {_, <<"200">>} = lists:keyfind(<<":status">>, 1, RespHeaders)
-
-
- after
- cowboy:stop_listener(?FUNCTION_NAME)
- end.
- settings_max_concurrent_streams(Config0) ->
- doc("The SETTINGS_MAX_CONCURRENT_STREAMS setting can be used to "
- "restrict the number of concurrent streams. (RFC7540 5.1.2, RFC7540 6.5.2)"),
-
- Config = cowboy_test:init_http(?FUNCTION_NAME, #{
- env => #{dispatch => cowboy_router:compile(init_routes(Config0))},
- max_concurrent_streams => 1
- }, Config0),
- try
- {ok, Socket} = do_handshake(Config),
-
- Headers = [
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ],
- {ReqHeadersBlock1, EncodeState} = cow_hpack:encode(Headers),
- {ReqHeadersBlock2, _} = cow_hpack:encode(Headers, EncodeState),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, fin, ReqHeadersBlock1),
- cow_http2:headers(3, fin, ReqHeadersBlock2)
- ]),
-
- {ok, << _:24, 3:8, _:8, 3:32, 7:32 >>} = gen_tcp:recv(Socket, 13, 6000)
- after
- cowboy:stop_listener(?FUNCTION_NAME)
- end.
- settings_max_concurrent_streams_0(Config0) ->
- doc("The SETTINGS_MAX_CONCURRENT_STREAMS setting can be set to "
- "0 to refuse all incoming streams. (RFC7540 5.1.2, RFC7540 6.5.2)"),
-
- Config = cowboy_test:init_http(?FUNCTION_NAME, #{
- env => #{dispatch => cowboy_router:compile(init_routes(Config0))},
- max_concurrent_streams => 0
- }, Config0),
- try
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 7:32 >>} = gen_tcp:recv(Socket, 13, 6000)
- after
- cowboy:stop_listener(?FUNCTION_NAME)
- end.
- settings_initial_window_size(Config0) ->
- doc("The SETTINGS_INITIAL_WINDOW_SIZE setting can be used to "
- "change the initial window size of streams. (RFC7540 6.5.2)"),
-
- Config = cowboy_test:init_http(?FUNCTION_NAME, #{
- env => #{dispatch => cowboy_router:compile(init_routes(Config0))},
- initial_connection_window_size => 100000,
- initial_stream_window_size => 100000
- }, Config0),
- try
-
-
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len1:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len1/binary >>} = gen_tcp:recv(Socket, 6 + Len1, 1000),
-
- ok = gen_tcp:send(Socket, cow_http2:settings_ack()),
-
- {ok, << 4:24, 8:8, 0:40, _:32 >>} = gen_tcp:recv(Socket, 13, 1000),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
-
- Headers = [
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ],
- {HeadersBlock, _} = cow_hpack:encode(Headers),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, fin, <<0:15000/unit:8>>)
- ]),
-
- {ok, << Len2:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Len2, 6000),
-
- {error, timeout} = gen_tcp:recv(Socket, 0, 1000)
- after
- cowboy:stop_listener(?FUNCTION_NAME)
- end.
- settings_initial_window_size_after_ack(Config0) ->
- doc("The SETTINGS_INITIAL_WINDOW_SIZE setting can be used to "
- "change the initial window size of streams. It is applied "
- "to all existing streams upon receipt of the SETTINGS ack. (RFC7540 6.5.2)"),
-
- Config = cowboy_test:init_http(?FUNCTION_NAME, #{
- env => #{dispatch => cowboy_router:compile(init_routes(Config0))},
- initial_stream_window_size => 0
- }, Config0),
- try
-
-
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len1:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len1/binary >>} = gen_tcp:recv(Socket, 6 + Len1, 1000),
-
-
-
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
-
- Headers = [
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ],
- {HeadersBlock, _} = cow_hpack:encode(Headers),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:settings_ack(),
- cow_http2:data(1, fin, <<0:32/unit:8>>)
- ]),
-
- {ok, << _:24, 3:8, _:8, 1:32, 3:32 >>} = gen_tcp:recv(Socket, 13, 6000)
- after
- cowboy:stop_listener(?FUNCTION_NAME)
- end.
- settings_initial_window_size_before_ack(Config0) ->
- doc("The SETTINGS_INITIAL_WINDOW_SIZE setting can be used to "
- "change the initial window size of streams. It is only "
- "applied upon receipt of the SETTINGS ack. (RFC7540 6.5.2)"),
-
- Config = cowboy_test:init_http(?FUNCTION_NAME, #{
- env => #{dispatch => cowboy_router:compile(init_routes(Config0))},
- initial_stream_window_size => 0
- }, Config0),
- try
-
-
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len1:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len1/binary >>} = gen_tcp:recv(Socket, 6 + Len1, 1000),
-
-
-
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
-
- Headers = [
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ],
- {HeadersBlock, _} = cow_hpack:encode(Headers),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, fin, <<0:15000/unit:8>>)
- ]),
-
- {ok, << Len2:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Len2, 6000),
-
- {error, timeout} = gen_tcp:recv(Socket, 0, 1000)
- after
- cowboy:stop_listener(?FUNCTION_NAME)
- end.
- settings_max_frame_size(Config0) ->
- doc("The SETTINGS_MAX_FRAME_SIZE setting can be used to "
- "change the maximum frame size allowed. (RFC7540 6.5.2)"),
-
- Config = cowboy_test:init_http(?FUNCTION_NAME, #{
- env => #{dispatch => cowboy_router:compile(init_routes(Config0))},
- max_frame_size_received => 30000
- }, Config0),
- try
-
- {ok, Socket} = do_handshake(Config),
-
-
- Headers = [
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ],
- {HeadersBlock, _} = cow_hpack:encode(Headers),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, fin, <<0:25000/unit:8>>)
- ]),
-
- {ok, << Len2:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, _} = gen_tcp:recv(Socket, Len2, 6000),
-
- {error, timeout} = gen_tcp:recv(Socket, 0, 1000)
- after
- cowboy:stop_listener(?FUNCTION_NAME)
- end.
- settings_max_frame_size_reject_too_small(Config) ->
- doc("A SETTINGS_MAX_FRAME_SIZE smaller than 16384 must be rejected "
- "with a PROTOCOL_ERROR connection error. (RFC7540 6.5.2)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 6:24, 4:8, 0:40, 5:16, 16383:32 >>),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- settings_max_frame_size_reject_too_large(Config) ->
- doc("A SETTINGS_MAX_FRAME_SIZE larger than 16777215 must be rejected "
- "with a PROTOCOL_ERROR connection error. (RFC7540 6.5.2)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, << 6:24, 4:8, 0:40, 5:16, 16777216:32 >>),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- client_settings_disable_push(Config) ->
- doc("PUSH_PROMISE frames must not be sent when the setting "
- "SETTINGS_ENABLE_PUSH is disabled. (RFC7540 6.5.2, RFC7540 6.6, RFC7540 8.2)"),
-
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{
- enable_push => false
- })]),
-
- {ok, << Len:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len/binary >>} = gen_tcp:recv(Socket, 6 + Len, 1000),
-
- ok = gen_tcp:send(Socket, cow_http2:settings_ack()),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/resp/push">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- graceful_shutdown_client_stays(Config) ->
- doc("A server gracefully shutting down must send a GOAWAY frame with the "
- "last stream identifier set to 2^31-1 and a NO_ERROR code. After allowing "
- "time for any in-flight stream creation the server can send another GOAWAY "
- "frame with an updated last stream identifier. (RFC7540 6.8)"),
- {ok, Socket} = do_handshake(Config),
- ServerConnPid = get_remote_pid_tcp(Socket),
- ok = sys:terminate(ServerConnPid, whatever),
-
- {ok, <<_:24, 7:8, 0:8, 0:1, 0:31, 0:1, 16#7fffffff:31, 0:32>>} = gen_tcp:recv(Socket, 17, 500),
-
- {ok, <<_:24, 7:8, 0:8, 0:1, 0:31, 0:1, 0:31, 0:32>>} = gen_tcp:recv(Socket, 17, 1500),
- {error, closed} = gen_tcp:recv(Socket, 3, 1000),
- ok.
- graceful_shutdown_race_condition(Config) ->
- doc("A server in the process of gracefully shutting down must discard frames "
- "for streams initiated by the receiver with identifiers higher than the "
- "identified last stream. This may include frames that alter connection "
- "state such as HEADERS frames. (RFC7540 6.8)"),
- {ok, Socket} = do_handshake(Config),
- ServerConnPid = get_remote_pid_tcp(Socket),
- ok = sys:terminate(ServerConnPid, whatever),
-
- {ok, <<_:24, 7:8, 0:8, 0:1, 0:31, 0:1, 16#7fffffff:31, 0:32>>} = gen_tcp:recv(Socket, 17, 500),
-
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/delay_hello">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, <<_:24, 7:8, 0:8, 0:1, 0:31, 0:1, 1:31, 0:32>>} = gen_tcp:recv(Socket, 17, 2000),
-
- ok = gen_tcp:send(Socket, cow_http2:headers(3, fin, HeadersBlock)),
-
- {ok, <<RespHeadersPayloadLength:24, 1, 4, 0:1, 1:31>>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _RespHeaders} = gen_tcp:recv(Socket, RespHeadersPayloadLength, 1000),
- {ok, <<12:24, 0, 1, 0:1, 1:31, "Hello world!">>} = gen_tcp:recv(Socket, 21, 1000),
- {error, closed} = gen_tcp:recv(Socket, 3, 1000),
- ok.
- window_update_reject_0(Config) ->
- doc("WINDOW_UPDATE frames with an increment of 0 for the connection "
- "flow control window must be rejected with a "
- "PROTOCOL_ERROR connection error. (RFC7540 6.9.1)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, [
- cow_http2:window_update(0)
- ]),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- window_update_reject_0_stream(Config) ->
- doc("WINDOW_UPDATE frames with an increment of 0 for a stream "
- "flow control window must be rejected with a "
- "PROTOCOL_ERROR stream error. (RFC7540 6.9.1)"),
- {ok, Socket} = do_handshake(Config),
-
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, fin, HeadersBlock),
- cow_http2:window_update(1, 0)
- ]),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- data_reject_overflow(Config0) ->
- doc("DATA frames that cause the connection flow control window "
- "to overflow must be rejected with a FLOW_CONTROL_ERROR "
- "connection error. (RFC7540 6.9.1)"),
-
- Config = cowboy_test:init_http(?FUNCTION_NAME, #{
- env => #{dispatch => cowboy_router:compile(init_routes(Config0))},
- initial_stream_window_size => 100000
- }, Config0),
- try
- {ok, Socket} = do_handshake(Config),
-
-
- Headers = [
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ],
- {HeadersBlock, _} = cow_hpack:encode(Headers),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, fin, <<0:15000/unit:8>>)
- ]),
-
- {ok, << _:24, 7:8, _:72, 3:32 >>} = gen_tcp:recv(Socket, 17, 6000)
- after
- cowboy:stop_listener(?FUNCTION_NAME)
- end.
- data_reject_overflow_stream(Config0) ->
- doc("DATA frames that cause the stream flow control window "
- "to overflow must be rejected with a FLOW_CONTROL_ERROR "
- "stream error. (RFC7540 6.9.1)"),
-
- Config = cowboy_test:init_http(?FUNCTION_NAME, #{
- env => #{dispatch => cowboy_router:compile(init_routes(Config0))},
- initial_connection_window_size => 100000
- }, Config0),
- try
-
-
- {ok, Socket} = gen_tcp:connect("localhost", config(port, Config), [binary, {active, false}]),
-
- ok = gen_tcp:send(Socket, ["PRI * HTTP/2.0\r\n\r\nSM\r\n\r\n", cow_http2:settings(#{})]),
-
- {ok, << Len1:24 >>} = gen_tcp:recv(Socket, 3, 1000),
- {ok, << 4:8, 0:40, _:Len1/binary >>} = gen_tcp:recv(Socket, 6 + Len1, 1000),
-
- ok = gen_tcp:send(Socket, cow_http2:settings_ack()),
-
- {ok, << 4:24, 8:8, 0:40, _:32 >>} = gen_tcp:recv(Socket, 13, 1000),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
-
- Headers = [
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ],
- {HeadersBlock, _} = cow_hpack:encode(Headers),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, nofin, <<0:15000/unit:8>>),
- cow_http2:data(1, fin, <<0:15000/unit:8>>)
- ]),
-
- {ok, << _:24, 3:8, _:8, 1:32, 3:32 >>} = gen_tcp:recv(Socket, 13, 6000)
- after
- cowboy:stop_listener(?FUNCTION_NAME)
- end.
- window_update_reject_overflow(Config) ->
- doc("WINDOW_UPDATE frames that cause the connection flow control "
- "window to exceed 2^31-1 must be rejected with a "
- "FLOW_CONTROL_ERROR connection error. (RFC7540 6.9.1)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, [
- cow_http2:window_update(16#7fffffff)
- ]),
-
- {ok, << _:24, 7:8, _:72, 3:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- window_update_reject_overflow_stream(Config) ->
- doc("WINDOW_UPDATE frames that cause a stream flow control "
- "window to exceed 2^31-1 must be rejected with a "
- "FLOW_CONTROL_ERROR stream error. (RFC7540 6.9.1)"),
- {ok, Socket} = do_handshake(Config),
-
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, fin, HeadersBlock),
- cow_http2:window_update(1, 16#7fffffff)
- ]),
-
- {ok, << _:24, 3:8, _:8, 1:32, 3:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- settings_initial_window_size_changes(Config) ->
- doc("When the value of SETTINGS_INITIAL_WINDOW_SIZE changes, the server "
- "must adjust the size of the flow control windows of the active "
- "streams. (RFC7540 6.9.2)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, cow_http2:settings(#{initial_window_size => 0})),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {error, timeout} = gen_tcp:recv(Socket, 9, 1000),
-
- ok = gen_tcp:send(Socket, cow_http2:settings(#{initial_window_size => 5})),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
- {ok, << 5:24, 0:8, 0:8, 1:32, "Hello" >>} = gen_tcp:recv(Socket, 14, 1000),
- {error, timeout} = gen_tcp:recv(Socket, 9, 1000),
-
- ok = gen_tcp:send(Socket, cow_http2:settings(#{initial_window_size => 12})),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
- {ok, << 7:24, 0:8, 1:8, 1:32, " world!" >>} = gen_tcp:recv(Socket, 16, 1000),
- ok.
- settings_initial_window_size_changes_negative(Config) ->
- doc("When the value of SETTINGS_INITIAL_WINDOW_SIZE changes, the server "
- "must adjust the size of the flow control windows of the active "
- "streams even if their window end up negative. (RFC7540 6.9.2)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, cow_http2:settings(#{initial_window_size => 5})),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << SkipLen:24, 1:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- {ok, _} = gen_tcp:recv(Socket, SkipLen, 1000),
- {ok, << 5:24, 0:8, 0:8, 1:32, "Hello" >>} = gen_tcp:recv(Socket, 14, 1000),
- {error, timeout} = gen_tcp:recv(Socket, 9, 1000),
-
- ok = gen_tcp:send(Socket, cow_http2:settings(#{initial_window_size => 0})),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
- ok = gen_tcp:send(Socket, cow_http2:settings(#{initial_window_size => 12})),
-
- {ok, << 0:24, 4:8, 1:8, 0:32 >>} = gen_tcp:recv(Socket, 9, 1000),
-
- {ok, << 7:24, 0:8, 1:8, 1:32, " world!" >>} = gen_tcp:recv(Socket, 16, 1000),
- ok.
- settings_initial_window_size_reject_overflow(Config) ->
- doc("A SETTINGS_INITIAL_WINDOW_SIZE that causes a flow control window "
- "to exceed 2^31-1 must be rejected with a FLOW_CONTROL_ERROR "
- "connection error. (RFC7540 6.5.2, RFC7540 6.9.2)"),
- {ok, Socket} = do_handshake(Config),
-
- ok = gen_tcp:send(Socket, cow_http2:settings(#{initial_window_size => 16#80000000})),
-
- {ok, << _:24, 7:8, _:72, 3:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- accept_trailers(Config) ->
- doc("Trailing HEADERS frames must be accepted. (RFC7540 8.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, EncodeState} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>},
- {<<"trailer">>, <<"x-checksum">>}
- ]),
- {TrailersBlock, _} = cow_hpack:encode([
- {<<"x-checksum">>, <<"md5:4cc909a007407f3706399b6496babec3">>}
- ], EncodeState),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, nofin, <<0:10000/unit:8>>),
- cow_http2:headers(1, fin, TrailersBlock)
- ]),
-
- {ok, << _:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- accept_trailers_continuation(Config) ->
- doc("Trailing HEADERS and CONTINUATION frames must be accepted. (RFC7540 8.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, EncodeState} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>},
- {<<"trailer">>, <<"x-checksum">>}
- ]),
- {TrailersBlock, _} = cow_hpack:encode([
- {<<"x-checksum">>, <<"md5:4cc909a007407f3706399b6496babec3">>}
- ], EncodeState),
- Len = iolist_size(TrailersBlock),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, nofin, <<0:10000/unit:8>>),
- <<0:24, 1:8, 0:7, 1:1, 0:1, 1:31>>,
- <<Len:24, 9:8, 0:5, 1:1, 0:3, 1:31>>,
- TrailersBlock
- ]),
-
- {ok, << _:24, 1:8, _:40 >>} = gen_tcp:recv(Socket, 9, 6000),
- ok.
- reject_trailers_nofin(Config) ->
- doc("Trailing HEADERS frames received without the END_STREAM flag "
- "set must be rejected with a PROTOCOL_ERROR connection error. "
- "(RFC7540 8.1, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
-
- {HeadersBlock, EncodeState} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>},
- {<<"trailer">>, <<"x-checksum">>}
- ]),
- {TrailersBlock, _} = cow_hpack:encode([
- {<<"x-checksum">>, <<"md5:4cc909a007407f3706399b6496babec3">>}
- ], EncodeState),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, nofin, <<0:10000/unit:8>>),
- cow_http2:headers(1, nofin, TrailersBlock)
- ]),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- reject_trailers_nofin_continuation(Config) ->
- doc("Trailing HEADERS frames received without the END_STREAM flag "
- "set must be rejected with a PROTOCOL_ERROR connection error. "
- "(RFC7540 8.1, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
-
- {HeadersBlock, EncodeState} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>},
- {<<"trailer">>, <<"x-checksum">>}
- ]),
- {TrailersBlock, _} = cow_hpack:encode([
- {<<"x-checksum">>, <<"md5:4cc909a007407f3706399b6496babec3">>}
- ], EncodeState),
- Len = iolist_size(TrailersBlock),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, nofin, <<0:10000/unit:8>>),
- <<0:24, 1:8, 0:9, 1:31>>,
- <<Len:24, 9:8, 0:5, 1:1, 0:3, 1:31>>,
- TrailersBlock
- ]),
-
- {ok, << _:24, 7:8, _:72, 1:32 >>} = gen_tcp:recv(Socket, 17, 6000),
- ok.
- headers_informational_nofin(Config) ->
- doc("Informational HEADERS frames must not have the END_STREAM flag set. (RFC7540 8.1)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/echo/read_body">>},
- {<<"expect">>, <<"100-continue">>},
- {<<"content-length">>, <<"1000000">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, nofin, HeadersBlock)),
-
- {ok, << Len:24, 1:8, 0:5, 1:1, 0:2, _:32 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, RespHeadersBlock} = gen_tcp:recv(Socket, Len, 6000),
-
- {RespHeaders, _} = cow_hpack:decode(RespHeadersBlock),
- {_, <<"100">>} = lists:keyfind(<<":status">>, 1, RespHeaders),
- ok.
- headers_reject_uppercase_header_name(Config) ->
- doc("Requests containing uppercase header names must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>},
- {<<"HELLO">>, <<"world">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_response_pseudo_headers(Config) ->
- doc("Requests containing response pseudo-headers must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.1, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>},
- {<<":status">>, <<"200">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_unknown_pseudo_headers(Config) ->
- doc("Requests containing unknown pseudo-headers must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.1, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>},
- {<<":upgrade">>, <<"websocket">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_pseudo_headers_in_trailers(Config) ->
- doc("Requests containing pseudo-headers in trailers must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.1, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
-
- {HeadersBlock, EncodeState} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>},
- {<<"trailer">>, <<"x-checksum">>}
- ]),
- {TrailersBlock, _} = cow_hpack:encode([
- {<<"x-checksum">>, <<"md5:4cc909a007407f3706399b6496babec3">>},
- {<<":path">>, <<"/">>}
- ], EncodeState),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, nofin, <<0:10000/unit:8>>),
- cow_http2:headers(1, fin, TrailersBlock)
- ]),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_pseudo_headers_after_regular_headers(Config) ->
- doc("Requests containing pseudo-headers after regular headers must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.1, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<"content-length">>, <<"0">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_connection_header(Config) ->
- doc("Requests containing a connection header must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.2, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>},
- {<<"connection">>, <<"close">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_keep_alive_header(Config) ->
- doc("Requests containing a keep-alive header must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.2, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>},
- {<<"keep-alive">>, <<"timeout=5, max=1000">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_proxy_authenticate_header(Config) ->
- doc("Requests containing a connection header must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.2, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>},
- {<<"proxy-authenticate">>, <<"Basic">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_proxy_authorization_header(Config) ->
- doc("Requests containing a connection header must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.2, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>},
- {<<"proxy-authorization">>, <<"Basic YWxhZGRpbjpvcGVuc2VzYW1l">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_transfer_encoding_header(Config) ->
- doc("Requests containing a connection header must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.2, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>},
- {<<"transfer-encoding">>, <<"chunked">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_upgrade_header(Config) ->
- doc("Requests containing a connection header must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.2, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>},
- {<<"upgrade">>, <<"websocket">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- accept_te_header_value_trailers(Config) ->
- doc("Requests containing a TE header with a value of \"trailers\" "
- "must be accepted. (RFC7540 8.1.2.2)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>},
- {<<"te">>, <<"trailers">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 1:8, _:8, 1:32 >>} = gen_tcp:recv(Socket, 9, 1000),
- ok.
- reject_te_header_other_values(Config) ->
- doc("Requests containing a TE header with a value other than \"trailers\" must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.2, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>},
- {<<"te">>, <<"trailers, deflate;q=0.5">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- response_dont_send_header_in_connection(Config) ->
- doc("Intermediaries must remove HTTP/1.1 connection headers when "
- "transforming an HTTP/1.1 messages to HTTP/2. The server must "
- "not send them either. All headers listed in the connection "
- "header must be removed. (RFC7540 8.1.2.2)"),
- do_response_dont_send_http11_header(Config, <<"custom-header">>).
- response_dont_send_connection_header(Config) ->
- doc("Intermediaries must remove HTTP/1.1 connection headers when "
- "transforming an HTTP/1.1 messages to HTTP/2. The server must "
- "not send them either. The connection header must be removed. (RFC7540 8.1.2.2)"),
- do_response_dont_send_http11_header(Config, <<"connection">>).
- response_dont_send_keep_alive_header(Config) ->
- doc("Intermediaries must remove HTTP/1.1 connection headers when "
- "transforming an HTTP/1.1 messages to HTTP/2. The server must "
- "not send them either. The keep-alive header must be removed "
- "even if not listed in the connection header. (RFC7540 8.1.2.2)"),
- do_response_dont_send_http11_header(Config, <<"keep-alive">>).
- response_dont_send_proxy_connection_header(Config) ->
- doc("Intermediaries must remove HTTP/1.1 connection headers when "
- "transforming an HTTP/1.1 messages to HTTP/2. The server must "
- "not send them either. The proxy-connection header must be removed "
- "even if not listed in the connection header. (RFC7540 8.1.2.2)"),
- do_response_dont_send_http11_header(Config, <<"proxy-connection">>).
- response_dont_send_transfer_encoding_header(Config) ->
- doc("Intermediaries must remove HTTP/1.1 connection headers when "
- "transforming an HTTP/1.1 messages to HTTP/2. The server must "
- "not send them either. The transfer-encoding header must be removed "
- "even if not listed in the connection header. (RFC7540 8.1.2.2)"),
- do_response_dont_send_http11_header(Config, <<"transfer-encoding">>).
- response_dont_send_upgrade_header(Config) ->
- doc("Intermediaries must remove HTTP/1.1 connection headers when "
- "transforming an HTTP/1.1 messages to HTTP/2. The server must "
- "not send them either. The upgrade header must be removed "
- "even if not listed in the connection header. (RFC7540 8.1.2.2)"),
- do_response_dont_send_http11_header(Config, <<"upgrade">>).
- do_response_dont_send_http11_header(Config, Name) ->
- ConnPid = gun_open(Config),
- Ref = gun:get(ConnPid, "/resp/set_resp_headers_http11"),
- {response, nofin, 200, Headers} = gun:await(ConnPid, Ref),
- false = lists:keyfind(Name, 1, Headers),
- ok.
- reject_userinfo(Config) ->
- doc("An authority containing a userinfo component must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.3, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"user@localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_empty_path(Config) ->
- doc("A request containing an empty path component must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.3, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<>>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_missing_pseudo_header_method(Config) ->
- doc("A request without a method component must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.3, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_many_pseudo_header_method(Config) ->
- doc("A request containing more than one method component must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.3, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_missing_pseudo_header_scheme(Config) ->
- doc("A request without a scheme component must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.3, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_many_pseudo_header_scheme(Config) ->
- doc("A request containing more than one scheme component must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.3, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_missing_pseudo_header_authority(Config) ->
- doc("A request without an authority or host component must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.3, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- accept_host_header_on_missing_pseudo_header_authority(Config) ->
- doc("A request without an authority but with a host header must be accepted. "
- "(RFC7540 8.1.2.3, RFC7540 8.1.3)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":path">>, <<"/">>},
- {<<"host">>, <<"localhost">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << Len:24, 1:8, _:8, _:32 >>} = gen_tcp:recv(Socket, 9, 6000),
- {ok, RespHeadersBlock} = gen_tcp:recv(Socket, Len, 6000),
- {RespHeaders, _} = cow_hpack:decode(RespHeadersBlock),
- {_, <<"200">>} = lists:keyfind(<<":status">>, 1, RespHeaders),
- ok.
- reject_many_pseudo_header_authority(Config) ->
- doc("A request containing more than one authority component must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.3, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_missing_pseudo_header_path(Config) ->
- doc("A request without a path component must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.3, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_many_pseudo_header_path(Config) ->
- doc("A request containing more than one path component must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.3, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>},
- {<<":path">>, <<"/">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_data_size_smaller_than_content_length(Config) ->
- doc("Requests that have a content-length header whose value does not "
- "match the total length of the DATA frames must be rejected with "
- "a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>},
- {<<"content-length">>, <<"12">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, fin, <<"Hello!">>)
- ]),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_data_size_larger_than_content_length(Config) ->
- doc("Requests that have a content-length header whose value does not "
- "match the total length of the DATA frames must be rejected with "
- "a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>},
- {<<"content-length">>, <<"12">>}
- ]),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, nofin, <<"Hello! World! Universe!">>),
- cow_http2:data(1, fin, <<"Multiverse!">>)
- ]),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_content_length_without_data(Config) ->
- doc("Requests that have a content-length header whose value does not "
- "match the total length of the DATA frames must be rejected with "
- "a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>},
- {<<"content-length">>, <<"12">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, fin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_data_size_different_than_content_length_with_trailers(Config) ->
- doc("Requests that have a content-length header whose value does not "
- "match the total length of the DATA frames must be rejected with "
- "a PROTOCOL_ERROR stream error. (RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
-
- {HeadersBlock, EncodeState} = cow_hpack:encode([
- {<<":method">>, <<"POST">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/long_polling">>},
- {<<"content-length">>, <<"12">>},
- {<<"trailer">>, <<"x-checksum">>}
- ]),
- {TrailersBlock, _} = cow_hpack:encode([
- {<<"x-checksum">>, <<"md5:4cc909a007407f3706399b6496babec3">>}
- ], EncodeState),
- ok = gen_tcp:send(Socket, [
- cow_http2:headers(1, nofin, HeadersBlock),
- cow_http2:data(1, nofin, <<"Hello!">>),
- cow_http2:headers(1, fin, TrailersBlock)
- ]),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- reject_duplicate_content_length_header(Config) ->
- doc("A request with duplicate content-length headers must be rejected "
- "with a PROTOCOL_ERROR stream error. (RFC7230 3.3.2, RFC7540 8.1.2.6)"),
- {ok, Socket} = do_handshake(Config),
-
- {HeadersBlock, _} = cow_hpack:encode([
- {<<":method">>, <<"GET">>},
- {<<":scheme">>, <<"http">>},
- {<<":authority">>, <<"localhost">>},
- {<<":path">>, <<"/">>},
- {<<"content-length">>, <<"12">>},
- {<<"content-length">>, <<"12">>}
- ]),
- ok = gen_tcp:send(Socket, cow_http2:headers(1, nofin, HeadersBlock)),
-
- {ok, << _:24, 3:8, _:8, 1:32, 1:32 >>} = gen_tcp:recv(Socket, 13, 6000),
- ok.
- push_has_no_request_body(Config) ->
- doc("PUSH_PROMISE frames include the complete set of request headers "
- "and the request can never include a body. (RFC7540 8.2.1)"),
- ConnPid = gun_open(Config),
- Ref = gun:get(ConnPid, "/resp/push/read_body"),
- {push, PushRef, <<"GET">>, _, _} = gun:await(ConnPid, Ref),
- {response, fin, 200, _} = gun:await(ConnPid, Ref),
-
-
- {response, fin, 200, _} = gun:await(ConnPid, PushRef),
- ok.
- status_code_421(Config) ->
- doc("The 421 Misdirected Request status code can be sent. (RFC7540 9.1.2)"),
- ConnPid = gun_open(Config),
- Ref = gun:get(ConnPid, "/resp/reply2/421"),
- {response, fin, 421, _} = gun:await(ConnPid, Ref),
- ok.
|