| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750 |
- <?php
- /**
- * CakePHP(tm) : Rapid Development Framework (https://cakephp.org)
- * Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
- *
- * Licensed under The MIT License
- * For full copyright and license information, please see the LICENSE.txt
- * Redistributions of files must retain the above copyright notice.
- *
- * @copyright Copyright (c) Cake Software Foundation, Inc. (https://cakefoundation.org)
- * @link https://cakephp.org CakePHP(tm) Project
- * @since 2.0.0
- * @license https://opensource.org/licenses/mit-license.php MIT License
- */
- namespace Cake\Http;
- use Cake\Core\Configure;
- use Cake\Filesystem\File;
- use Cake\Http\Cookie\Cookie;
- use Cake\Http\Cookie\CookieCollection;
- use Cake\Http\Cookie\CookieInterface;
- use Cake\Http\CorsBuilder;
- use Cake\Http\Exception\NotFoundException;
- use Cake\Log\Log;
- use DateTime;
- use DateTimeZone;
- use InvalidArgumentException;
- use Psr\Http\Message\ResponseInterface;
- use Psr\Http\Message\StreamInterface;
- use Zend\Diactoros\MessageTrait;
- use Zend\Diactoros\Stream;
- /**
- * Responses contain the response text, status and headers of a HTTP response.
- */
- class Response implements ResponseInterface
- {
- use MessageTrait;
- /**
- * Holds HTTP response statuses
- *
- * @var array
- */
- protected $_statusCodes = [
- 100 => 'Continue',
- 101 => 'Switching Protocols',
- 102 => 'Processing',
- 200 => 'OK',
- 201 => 'Created',
- 202 => 'Accepted',
- 203 => 'Non-Authoritative Information',
- 204 => 'No Content',
- 205 => 'Reset Content',
- 206 => 'Partial Content',
- 207 => 'Multi-status',
- 208 => 'Already Reported',
- 226 => 'IM used',
- 300 => 'Multiple Choices',
- 301 => 'Moved Permanently',
- 302 => 'Found',
- 303 => 'See Other',
- 304 => 'Not Modified',
- 305 => 'Use Proxy',
- 306 => '(Unused)',
- 307 => 'Temporary Redirect',
- 308 => 'Permanent Redirect',
- 400 => 'Bad Request',
- 401 => 'Unauthorized',
- 402 => 'Payment Required',
- 403 => 'Forbidden',
- 404 => 'Not Found',
- 405 => 'Method Not Allowed',
- 406 => 'Not Acceptable',
- 407 => 'Proxy Authentication Required',
- 408 => 'Request Timeout',
- 409 => 'Conflict',
- 410 => 'Gone',
- 411 => 'Length Required',
- 412 => 'Precondition Failed',
- 413 => 'Request Entity Too Large',
- 414 => 'Request-URI Too Large',
- 415 => 'Unsupported Media Type',
- 416 => 'Requested range not satisfiable',
- 417 => 'Expectation Failed',
- 418 => 'I\'m a teapot',
- 421 => 'Misdirected Request',
- 422 => 'Unprocessable Entity',
- 423 => 'Locked',
- 424 => 'Failed Dependency',
- 425 => 'Unordered Collection',
- 426 => 'Upgrade Required',
- 428 => 'Precondition Required',
- 429 => 'Too Many Requests',
- 431 => 'Request Header Fields Too Large',
- 444 => 'Connection Closed Without Response',
- 451 => 'Unavailable For Legal Reasons',
- 499 => 'Client Closed Request',
- 500 => 'Internal Server Error',
- 501 => 'Not Implemented',
- 502 => 'Bad Gateway',
- 503 => 'Service Unavailable',
- 504 => 'Gateway Timeout',
- 505 => 'Unsupported Version',
- 506 => 'Variant Also Negotiates',
- 507 => 'Insufficient Storage',
- 508 => 'Loop Detected',
- 510 => 'Not Extended',
- 511 => 'Network Authentication Required',
- 599 => 'Network Connect Timeout Error',
- ];
- /**
- * Holds type key to mime type mappings for known mime types.
- *
- * @var array
- */
- protected $_mimeTypes = [
- 'html' => ['text/html', '*/*'],
- 'json' => 'application/json',
- 'xml' => ['application/xml', 'text/xml'],
- 'xhtml' => ['application/xhtml+xml', 'application/xhtml', 'text/xhtml'],
- 'webp' => 'image/webp',
- 'rss' => 'application/rss+xml',
- 'ai' => 'application/postscript',
- 'bcpio' => 'application/x-bcpio',
- 'bin' => 'application/octet-stream',
- 'ccad' => 'application/clariscad',
- 'cdf' => 'application/x-netcdf',
- 'class' => 'application/octet-stream',
- 'cpio' => 'application/x-cpio',
- 'cpt' => 'application/mac-compactpro',
- 'csh' => 'application/x-csh',
- 'csv' => ['text/csv', 'application/vnd.ms-excel'],
- 'dcr' => 'application/x-director',
- 'dir' => 'application/x-director',
- 'dms' => 'application/octet-stream',
- 'doc' => 'application/msword',
- 'docx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
- 'drw' => 'application/drafting',
- 'dvi' => 'application/x-dvi',
- 'dwg' => 'application/acad',
- 'dxf' => 'application/dxf',
- 'dxr' => 'application/x-director',
- 'eot' => 'application/vnd.ms-fontobject',
- 'eps' => 'application/postscript',
- 'exe' => 'application/octet-stream',
- 'ez' => 'application/andrew-inset',
- 'flv' => 'video/x-flv',
- 'gtar' => 'application/x-gtar',
- 'gz' => 'application/x-gzip',
- 'bz2' => 'application/x-bzip',
- '7z' => 'application/x-7z-compressed',
- 'hdf' => 'application/x-hdf',
- 'hqx' => 'application/mac-binhex40',
- 'ico' => 'image/x-icon',
- 'ips' => 'application/x-ipscript',
- 'ipx' => 'application/x-ipix',
- 'js' => 'application/javascript',
- 'jsonapi' => 'application/vnd.api+json',
- 'latex' => 'application/x-latex',
- 'lha' => 'application/octet-stream',
- 'lsp' => 'application/x-lisp',
- 'lzh' => 'application/octet-stream',
- 'man' => 'application/x-troff-man',
- 'me' => 'application/x-troff-me',
- 'mif' => 'application/vnd.mif',
- 'ms' => 'application/x-troff-ms',
- 'nc' => 'application/x-netcdf',
- 'oda' => 'application/oda',
- 'otf' => 'font/otf',
- 'pdf' => 'application/pdf',
- 'pgn' => 'application/x-chess-pgn',
- 'pot' => 'application/vnd.ms-powerpoint',
- 'pps' => 'application/vnd.ms-powerpoint',
- 'ppt' => 'application/vnd.ms-powerpoint',
- 'pptx' => 'application/vnd.openxmlformats-officedocument.presentationml.presentation',
- 'ppz' => 'application/vnd.ms-powerpoint',
- 'pre' => 'application/x-freelance',
- 'prt' => 'application/pro_eng',
- 'ps' => 'application/postscript',
- 'roff' => 'application/x-troff',
- 'scm' => 'application/x-lotusscreencam',
- 'set' => 'application/set',
- 'sh' => 'application/x-sh',
- 'shar' => 'application/x-shar',
- 'sit' => 'application/x-stuffit',
- 'skd' => 'application/x-koan',
- 'skm' => 'application/x-koan',
- 'skp' => 'application/x-koan',
- 'skt' => 'application/x-koan',
- 'smi' => 'application/smil',
- 'smil' => 'application/smil',
- 'sol' => 'application/solids',
- 'spl' => 'application/x-futuresplash',
- 'src' => 'application/x-wais-source',
- 'step' => 'application/STEP',
- 'stl' => 'application/SLA',
- 'stp' => 'application/STEP',
- 'sv4cpio' => 'application/x-sv4cpio',
- 'sv4crc' => 'application/x-sv4crc',
- 'svg' => 'image/svg+xml',
- 'svgz' => 'image/svg+xml',
- 'swf' => 'application/x-shockwave-flash',
- 't' => 'application/x-troff',
- 'tar' => 'application/x-tar',
- 'tcl' => 'application/x-tcl',
- 'tex' => 'application/x-tex',
- 'texi' => 'application/x-texinfo',
- 'texinfo' => 'application/x-texinfo',
- 'tr' => 'application/x-troff',
- 'tsp' => 'application/dsptype',
- 'ttc' => 'font/ttf',
- 'ttf' => 'font/ttf',
- 'unv' => 'application/i-deas',
- 'ustar' => 'application/x-ustar',
- 'vcd' => 'application/x-cdlink',
- 'vda' => 'application/vda',
- 'xlc' => 'application/vnd.ms-excel',
- 'xll' => 'application/vnd.ms-excel',
- 'xlm' => 'application/vnd.ms-excel',
- 'xls' => 'application/vnd.ms-excel',
- 'xlsx' => 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
- 'xlw' => 'application/vnd.ms-excel',
- 'zip' => 'application/zip',
- 'aif' => 'audio/x-aiff',
- 'aifc' => 'audio/x-aiff',
- 'aiff' => 'audio/x-aiff',
- 'au' => 'audio/basic',
- 'kar' => 'audio/midi',
- 'mid' => 'audio/midi',
- 'midi' => 'audio/midi',
- 'mp2' => 'audio/mpeg',
- 'mp3' => 'audio/mpeg',
- 'mpga' => 'audio/mpeg',
- 'ogg' => 'audio/ogg',
- 'oga' => 'audio/ogg',
- 'spx' => 'audio/ogg',
- 'ra' => 'audio/x-realaudio',
- 'ram' => 'audio/x-pn-realaudio',
- 'rm' => 'audio/x-pn-realaudio',
- 'rpm' => 'audio/x-pn-realaudio-plugin',
- 'snd' => 'audio/basic',
- 'tsi' => 'audio/TSP-audio',
- 'wav' => 'audio/x-wav',
- 'aac' => 'audio/aac',
- 'asc' => 'text/plain',
- 'c' => 'text/plain',
- 'cc' => 'text/plain',
- 'css' => 'text/css',
- 'etx' => 'text/x-setext',
- 'f' => 'text/plain',
- 'f90' => 'text/plain',
- 'h' => 'text/plain',
- 'hh' => 'text/plain',
- 'htm' => ['text/html', '*/*'],
- 'ics' => 'text/calendar',
- 'm' => 'text/plain',
- 'rtf' => 'text/rtf',
- 'rtx' => 'text/richtext',
- 'sgm' => 'text/sgml',
- 'sgml' => 'text/sgml',
- 'tsv' => 'text/tab-separated-values',
- 'tpl' => 'text/template',
- 'txt' => 'text/plain',
- 'text' => 'text/plain',
- 'avi' => 'video/x-msvideo',
- 'fli' => 'video/x-fli',
- 'mov' => 'video/quicktime',
- 'movie' => 'video/x-sgi-movie',
- 'mpe' => 'video/mpeg',
- 'mpeg' => 'video/mpeg',
- 'mpg' => 'video/mpeg',
- 'qt' => 'video/quicktime',
- 'viv' => 'video/vnd.vivo',
- 'vivo' => 'video/vnd.vivo',
- 'ogv' => 'video/ogg',
- 'webm' => 'video/webm',
- 'mp4' => 'video/mp4',
- 'm4v' => 'video/mp4',
- 'f4v' => 'video/mp4',
- 'f4p' => 'video/mp4',
- 'm4a' => 'audio/mp4',
- 'f4a' => 'audio/mp4',
- 'f4b' => 'audio/mp4',
- 'gif' => 'image/gif',
- 'ief' => 'image/ief',
- 'jpg' => 'image/jpeg',
- 'jpeg' => 'image/jpeg',
- 'jpe' => 'image/jpeg',
- 'pbm' => 'image/x-portable-bitmap',
- 'pgm' => 'image/x-portable-graymap',
- 'png' => 'image/png',
- 'pnm' => 'image/x-portable-anymap',
- 'ppm' => 'image/x-portable-pixmap',
- 'ras' => 'image/cmu-raster',
- 'rgb' => 'image/x-rgb',
- 'tif' => 'image/tiff',
- 'tiff' => 'image/tiff',
- 'xbm' => 'image/x-xbitmap',
- 'xpm' => 'image/x-xpixmap',
- 'xwd' => 'image/x-xwindowdump',
- 'psd' => ['application/photoshop', 'application/psd', 'image/psd', 'image/x-photoshop', 'image/photoshop', 'zz-application/zz-winassoc-psd'],
- 'ice' => 'x-conference/x-cooltalk',
- 'iges' => 'model/iges',
- 'igs' => 'model/iges',
- 'mesh' => 'model/mesh',
- 'msh' => 'model/mesh',
- 'silo' => 'model/mesh',
- 'vrml' => 'model/vrml',
- 'wrl' => 'model/vrml',
- 'mime' => 'www/mime',
- 'pdb' => 'chemical/x-pdb',
- 'xyz' => 'chemical/x-pdb',
- 'javascript' => 'application/javascript',
- 'form' => 'application/x-www-form-urlencoded',
- 'file' => 'multipart/form-data',
- 'xhtml-mobile' => 'application/vnd.wap.xhtml+xml',
- 'atom' => 'application/atom+xml',
- 'amf' => 'application/x-amf',
- 'wap' => ['text/vnd.wap.wml', 'text/vnd.wap.wmlscript', 'image/vnd.wap.wbmp'],
- 'wml' => 'text/vnd.wap.wml',
- 'wmlscript' => 'text/vnd.wap.wmlscript',
- 'wbmp' => 'image/vnd.wap.wbmp',
- 'woff' => 'application/x-font-woff',
- 'appcache' => 'text/cache-manifest',
- 'manifest' => 'text/cache-manifest',
- 'htc' => 'text/x-component',
- 'rdf' => 'application/xml',
- 'crx' => 'application/x-chrome-extension',
- 'oex' => 'application/x-opera-extension',
- 'xpi' => 'application/x-xpinstall',
- 'safariextz' => 'application/octet-stream',
- 'webapp' => 'application/x-web-app-manifest+json',
- 'vcf' => 'text/x-vcard',
- 'vtt' => 'text/vtt',
- 'mkv' => 'video/x-matroska',
- 'pkpass' => 'application/vnd.apple.pkpass',
- 'ajax' => 'text/html'
- ];
- /**
- * Protocol header to send to the client
- *
- * @var string
- */
- protected $_protocol = 'HTTP/1.1';
- /**
- * Status code to send to the client
- *
- * @var int
- */
- protected $_status = 200;
- /**
- * Content type to send. This can be an 'extension' that will be transformed using the $_mimetypes array
- * or a complete mime-type
- *
- * @var string
- */
- protected $_contentType = 'text/html';
- /**
- * File object for file to be read out as response
- *
- * @var \Cake\Filesystem\File|null
- */
- protected $_file;
- /**
- * File range. Used for requesting ranges of files.
- *
- * @var array
- */
- protected $_fileRange = [];
- /**
- * The charset the response body is encoded with
- *
- * @var string
- */
- protected $_charset = 'UTF-8';
- /**
- * Holds all the cache directives that will be converted
- * into headers when sending the request
- *
- * @var array
- */
- protected $_cacheDirectives = [];
- /**
- * Collection of cookies to send to the client
- *
- * @var \Cake\Http\Cookie\CookieCollection
- */
- protected $_cookies = null;
- /**
- * Reason Phrase
- *
- * @var string
- */
- protected $_reasonPhrase = 'OK';
- /**
- * Stream mode options.
- *
- * @var string
- */
- protected $_streamMode = 'wb+';
- /**
- * Stream target or resource object.
- *
- * @var string|resource
- */
- protected $_streamTarget = 'php://memory';
- /**
- * Constructor
- *
- * @param array $options list of parameters to setup the response. Possible values are:
- * - body: the response text that should be sent to the client
- * - statusCodes: additional allowable response codes
- * - status: the HTTP status code to respond with
- * - type: a complete mime-type string or an extension mapped in this class
- * - charset: the charset for the response body
- */
- public function __construct(array $options = [])
- {
- if (isset($options['streamTarget'])) {
- $this->_streamTarget = $options['streamTarget'];
- }
- if (isset($options['streamMode'])) {
- $this->_streamMode = $options['streamMode'];
- }
- if (isset($options['stream'])) {
- if (!$options['stream'] instanceof StreamInterface) {
- throw new InvalidArgumentException('Stream option must be an object that implements StreamInterface');
- }
- $this->stream = $options['stream'];
- } else {
- $this->_createStream();
- }
- if (isset($options['body'])) {
- $this->stream->write($options['body']);
- }
- if (isset($options['statusCodes'])) {
- $this->httpCodes($options['statusCodes']);
- }
- if (isset($options['status'])) {
- $this->_setStatus($options['status']);
- }
- if (!isset($options['charset'])) {
- $options['charset'] = Configure::read('App.encoding');
- }
- $this->_charset = $options['charset'];
- if (isset($options['type'])) {
- $this->_contentType = $this->resolveType($options['type']);
- }
- $this->_setContentType();
- $this->_cookies = new CookieCollection();
- }
- /**
- * Creates the stream object.
- *
- * @return void
- */
- protected function _createStream()
- {
- $this->stream = new Stream($this->_streamTarget, $this->_streamMode);
- }
- /**
- * Sends the complete response to the client including headers and message body.
- * Will echo out the content in the response body.
- *
- * @return void
- * @deprecated 3.4.0 Will be removed in 4.0.0
- */
- public function send()
- {
- deprecationWarning('Response::send() will be removed in 4.0.0');
- if ($this->hasHeader('Location') && $this->_status === 200) {
- $this->statusCode(302);
- }
- $this->_setContent();
- $this->sendHeaders();
- if ($this->_file) {
- $this->_sendFile($this->_file, $this->_fileRange);
- $this->_file = null;
- $this->_fileRange = [];
- } else {
- $this->_sendContent($this->body());
- }
- if (function_exists('fastcgi_finish_request')) {
- fastcgi_finish_request();
- }
- }
- /**
- * Sends the HTTP headers and cookies.
- *
- * @return void
- * @deprecated 3.4.0 Will be removed in 4.0.0
- */
- public function sendHeaders()
- {
- deprecationWarning(
- 'Will be removed in 4.0.0'
- );
- $file = $line = null;
- if (headers_sent($file, $line)) {
- Log::warning("Headers already sent in {$file}:{$line}");
- return;
- }
- $codeMessage = $this->_statusCodes[$this->_status];
- $this->_setCookies();
- $this->_sendHeader("{$this->_protocol} {$this->_status} {$codeMessage}");
- $this->_setContentType();
- foreach ($this->headers as $header => $values) {
- foreach ((array)$values as $value) {
- $this->_sendHeader($header, $value);
- }
- }
- }
- /**
- * Sets the cookies that have been added via Cake\Http\Response::cookie() before any
- * other output is sent to the client. Will set the cookies in the order they
- * have been set.
- *
- * @return void
- * @deprecated 3.4.0 Will be removed in 4.0.0
- */
- protected function _setCookies()
- {
- deprecationWarning(
- 'Will be removed in 4.0.0'
- );
- foreach ($this->_cookies as $cookie) {
- setcookie(
- $cookie->getName(),
- $cookie->getValue(),
- $cookie->getExpiresTimestamp(),
- $cookie->getPath(),
- $cookie->getDomain(),
- $cookie->isSecure(),
- $cookie->isHttpOnly()
- );
- }
- }
- /**
- * Formats the Content-Type header based on the configured contentType and charset
- * the charset will only be set in the header if the response is of type text/*
- *
- * @return void
- */
- protected function _setContentType()
- {
- if (in_array($this->_status, [304, 204])) {
- $this->_clearHeader('Content-Type');
- return;
- }
- $whitelist = [
- 'application/javascript', 'application/json', 'application/xml', 'application/rss+xml'
- ];
- $charset = false;
- if ($this->_charset &&
- (strpos($this->_contentType, 'text/') === 0 || in_array($this->_contentType, $whitelist))
- ) {
- $charset = true;
- }
- if ($charset) {
- $this->_setHeader('Content-Type', "{$this->_contentType}; charset={$this->_charset}");
- } else {
- $this->_setHeader('Content-Type', "{$this->_contentType}");
- }
- }
- /**
- * Sets the response body to an empty text if the status code is 204 or 304
- *
- * @return void
- * @deprecated 3.4.0 Will be removed in 4.0.0
- */
- protected function _setContent()
- {
- deprecationWarning(
- 'Will be removed in 4.0.0'
- );
- if (in_array($this->_status, [304, 204])) {
- $this->body('');
- }
- }
- /**
- * Sends a header to the client.
- *
- * @param string $name the header name
- * @param string|null $value the header value
- * @return void
- * @deprecated 3.4.0 Will be removed in 4.0.0
- */
- protected function _sendHeader($name, $value = null)
- {
- deprecationWarning(
- 'Will be removed in 4.0.0'
- );
- if ($value === null) {
- header($name);
- } else {
- header("{$name}: {$value}");
- }
- }
- /**
- * Sends a content string to the client.
- *
- * If the content is a callable, it is invoked. The callable should either
- * return a string or output content directly and have no return value.
- *
- * @param string|callable $content String to send as response body or callable
- * which returns/outputs content.
- * @return void
- * @deprecated 3.4.0 Will be removed in 4.0.0
- */
- protected function _sendContent($content)
- {
- deprecationWarning(
- 'Will be removed in 4.0.0'
- );
- if (!is_string($content) && is_callable($content)) {
- $content = $content();
- }
- echo $content;
- }
- /**
- * Buffers a header string to be sent
- * Returns the complete list of buffered headers
- *
- * ### Single header
- * ```
- * header('Location', 'http://example.com');
- * ```
- *
- * ### Multiple headers
- * ```
- * header(['Location' => 'http://example.com', 'X-Extra' => 'My header']);
- * ```
- *
- * ### String header
- * ```
- * header('WWW-Authenticate: Negotiate');
- * ```
- *
- * ### Array of string headers
- * ```
- * header(['WWW-Authenticate: Negotiate', 'Content-type: application/pdf']);
- * ```
- *
- * Multiple calls for setting the same header name will have the same effect as setting the header once
- * with the last value sent for it
- * ```
- * header('WWW-Authenticate: Negotiate');
- * header('WWW-Authenticate: Not-Negotiate');
- * ```
- * will have the same effect as only doing
- * ```
- * header('WWW-Authenticate: Not-Negotiate');
- * ```
- *
- * @param string|array|null $header An array of header strings or a single header string
- * - an associative array of "header name" => "header value" is also accepted
- * - an array of string headers is also accepted
- * @param string|array|null $value The header value(s)
- * @return array List of headers to be sent
- * @deprecated 3.4.0 Use `withHeader()`, `getHeaderLine()` and `getHeaders()` instead.
- */
- public function header($header = null, $value = null)
- {
- deprecationWarning(
- 'Response::header() is deprecated. ' .
- 'Use `withHeader()`, `getHeaderLine()` and `getHeaders()` instead.'
- );
- if ($header === null) {
- return $this->getSimpleHeaders();
- }
- $headers = is_array($header) ? $header : [$header => $value];
- foreach ($headers as $header => $value) {
- if (is_numeric($header)) {
- list($header, $value) = [$value, null];
- }
- if ($value === null) {
- list($header, $value) = explode(':', $header, 2);
- }
- $lower = strtolower($header);
- if (array_key_exists($lower, $this->headerNames)) {
- $header = $this->headerNames[$lower];
- } else {
- $this->headerNames[$lower] = $header;
- }
- $this->headers[$header] = is_array($value) ? array_map('trim', $value) : [trim($value)];
- }
- return $this->getSimpleHeaders();
- }
- /**
- * Backwards compatibility helper for getting flattened headers.
- *
- * Previously CakePHP would store headers as a simple dictionary, now that
- * we're supporting PSR7, the internal storage has each header as an array.
- *
- * @return array
- */
- protected function getSimpleHeaders()
- {
- $out = [];
- foreach ($this->headers as $key => $values) {
- $header = $this->headerNames[strtolower($key)];
- if (count($values) === 1) {
- $values = $values[0];
- }
- $out[$header] = $values;
- }
- return $out;
- }
- /**
- * Accessor for the location header.
- *
- * Get/Set the Location header value.
- *
- * @param null|string $url Either null to get the current location, or a string to set one.
- * @return string|null When setting the location null will be returned. When reading the location
- * a string of the current location header value (if any) will be returned.
- * @deprecated 3.4.0 Mutable responses are deprecated. Use `withLocation()` and `getHeaderLine()`
- * instead.
- */
- public function location($url = null)
- {
- deprecationWarning(
- 'Response::location() is deprecated. ' .
- 'Mutable responses are deprecated. Use `withLocation()` and `getHeaderLine()` instead.'
- );
- if ($url === null) {
- $result = $this->getHeaderLine('Location');
- if (!$result) {
- return null;
- }
- return $result;
- }
- if ($this->_status === 200) {
- $this->_status = 302;
- }
- $this->_setHeader('Location', $url);
- return null;
- }
- /**
- * Return an instance with an updated location header.
- *
- * If the current status code is 200, it will be replaced
- * with 302.
- *
- * @param string $url The location to redirect to.
- * @return static A new response with the Location header set.
- */
- public function withLocation($url)
- {
- $new = $this->withHeader('Location', $url);
- if ($new->_status === 200) {
- $new->_status = 302;
- }
- return $new;
- }
- /**
- * Sets a header.
- *
- * @param string $header Header key.
- * @param string $value Header value.
- * @return void
- */
- protected function _setHeader($header, $value)
- {
- $normalized = strtolower($header);
- $this->headerNames[$normalized] = $header;
- $this->headers[$header] = [$value];
- }
- /**
- * Clear header
- *
- * @param string $header Header key.
- * @return void
- */
- protected function _clearHeader($header)
- {
- $normalized = strtolower($header);
- if (!isset($this->headerNames[$normalized])) {
- return;
- }
- $original = $this->headerNames[$normalized];
- unset($this->headerNames[$normalized], $this->headers[$original]);
- }
- /**
- * Buffers the response message to be sent
- * if $content is null the current buffer is returned
- *
- * @param string|callable|null $content the string or callable message to be sent
- * @return string|null Current message buffer if $content param is passed as null
- * @deprecated 3.4.0 Mutable response methods are deprecated. Use `withBody()` and `getBody()` instead.
- */
- public function body($content = null)
- {
- deprecationWarning(
- 'Response::body() is deprecated. ' .
- 'Mutable response methods are deprecated. Use `withBody()` and `getBody()` instead.'
- );
- if ($content === null) {
- if ($this->stream->isSeekable()) {
- $this->stream->rewind();
- }
- $result = $this->stream->getContents();
- if (strlen($result) === 0) {
- return null;
- }
- return $result;
- }
- // Compatibility with closure/streaming responses
- if (!is_string($content) && is_callable($content)) {
- $this->stream = new CallbackStream($content);
- } else {
- $this->_createStream();
- $this->stream->write($content);
- }
- return $content;
- }
- /**
- * Handles the callable body for backward compatibility reasons.
- *
- * @param callable $content Callable content.
- * @return string
- */
- protected function _handleCallableBody(callable $content)
- {
- ob_start();
- $result1 = $content();
- $result2 = ob_get_contents();
- ob_get_clean();
- if ($result1) {
- return $result1;
- }
- return $result2;
- }
- /**
- * Sets the HTTP status code to be sent
- * if $code is null the current code is returned
- *
- * If the status code is 304 or 204, the existing Content-Type header
- * will be cleared, as these response codes have no body.
- *
- * @param int|null $code the HTTP status code
- * @return int Current status code
- * @throws \InvalidArgumentException When an unknown status code is reached.
- * @deprecated 3.4.0 Use `getStatusCode()` and `withStatus()` instead.
- */
- public function statusCode($code = null)
- {
- deprecationWarning(
- 'Response::statusCode() is deprecated. ' .
- 'Use `getStatusCode()` and `withStatus()` instead.'
- );
- if ($code === null) {
- return $this->_status;
- }
- if (!isset($this->_statusCodes[$code])) {
- throw new InvalidArgumentException('Unknown status code');
- }
- $this->_setStatus($code);
- return $code;
- }
- /**
- * Gets the response status code.
- *
- * The status code is a 3-digit integer result code of the server's attempt
- * to understand and satisfy the request.
- *
- * @return int Status code.
- */
- public function getStatusCode()
- {
- return $this->_status;
- }
- /**
- * Return an instance with the specified status code and, optionally, reason phrase.
- *
- * If no reason phrase is specified, implementations MAY choose to default
- * to the RFC 7231 or IANA recommended reason phrase for the response's
- * status code.
- *
- * This method MUST be implemented in such a way as to retain the
- * immutability of the message, and MUST return an instance that has the
- * updated status and reason phrase.
- *
- * If the status code is 304 or 204, the existing Content-Type header
- * will be cleared, as these response codes have no body.
- *
- * @link https://tools.ietf.org/html/rfc7231#section-6
- * @link https://www.iana.org/assignments/http-status-codes/http-status-codes.xhtml
- * @param int $code The 3-digit integer result code to set.
- * @param string $reasonPhrase The reason phrase to use with the
- * provided status code; if none is provided, implementations MAY
- * use the defaults as suggested in the HTTP specification.
- * @return static
- * @throws \InvalidArgumentException For invalid status code arguments.
- */
- public function withStatus($code, $reasonPhrase = '')
- {
- $new = clone $this;
- $new->_setStatus($code, $reasonPhrase);
- return $new;
- }
- /**
- * Modifier for response status
- *
- * @param int $code The code to set.
- * @param string $reasonPhrase The response reason phrase.
- * @return void
- */
- protected function _setStatus($code, $reasonPhrase = '')
- {
- $this->_status = $code;
- if (empty($reasonPhrase) && isset($this->_statusCodes[$code])) {
- $reasonPhrase = $this->_statusCodes[$code];
- }
- $this->_reasonPhrase = $reasonPhrase;
- $this->_setContentType();
- }
- /**
- * Gets the response reason phrase associated with the status code.
- *
- * Because a reason phrase is not a required element in a response
- * status line, the reason phrase value MAY be null. Implementations MAY
- * choose to return the default RFC 7231 recommended reason phrase (or those
- * listed in the IANA HTTP Status Code Registry) for the response's
- * status code.
- *
- * @link https://tools.ietf.org/html/rfc7231#section-6
- * @link http://www.iana.org/assignments/http-status-codes/http-status-codes.xhtml
- * @return string Reason phrase; must return an empty string if none present.
- */
- public function getReasonPhrase()
- {
- return $this->_reasonPhrase;
- }
- /**
- * Queries & sets valid HTTP response codes & messages.
- *
- * @param int|array|null $code If $code is an integer, then the corresponding code/message is
- * returned if it exists, null if it does not exist. If $code is an array, then the
- * keys are used as codes and the values as messages to add to the default HTTP
- * codes. The codes must be integers greater than 99 and less than 1000. Keep in
- * mind that the HTTP specification outlines that status codes begin with a digit
- * between 1 and 5, which defines the class of response the client is to expect.
- * Example:
- *
- * httpCodes(404); // returns [404 => 'Not Found']
- *
- * httpCodes([
- * 381 => 'Unicorn Moved',
- * 555 => 'Unexpected Minotaur'
- * ]); // sets these new values, and returns true
- *
- * httpCodes([
- * 0 => 'Nothing Here',
- * -1 => 'Reverse Infinity',
- * 12345 => 'Universal Password',
- * 'Hello' => 'World'
- * ]); // throws an exception due to invalid codes
- *
- * For more on HTTP status codes see: http://www.w3.org/Protocols/rfc2616/rfc2616-sec6.html#sec6.1
- *
- * @return mixed Associative array of the HTTP codes as keys, and the message
- * strings as values, or null of the given $code does not exist.
- * @throws \InvalidArgumentException If an attempt is made to add an invalid status code
- * @deprecated 3.4.0 Will be removed in 4.0.0
- */
- public function httpCodes($code = null)
- {
- deprecationWarning('Response::httpCodes(). Will be removed in 4.0.0');
- if (empty($code)) {
- return $this->_statusCodes;
- }
- if (is_array($code)) {
- $codes = array_keys($code);
- $min = min($codes);
- if (!is_int($min) || $min < 100 || max($codes) > 999) {
- throw new InvalidArgumentException('Invalid status code');
- }
- $this->_statusCodes = $code + $this->_statusCodes;
- return true;
- }
- if (!isset($this->_statusCodes[$code])) {
- return null;
- }
- return [$code => $this->_statusCodes[$code]];
- }
- /**
- * Sets the response content type. It can be either a file extension
- * which will be mapped internally to a mime-type or a string representing a mime-type
- * if $contentType is null the current content type is returned
- * if $contentType is an associative array, content type definitions will be stored/replaced
- *
- * ### Setting the content type
- *
- * ```
- * type('jpg');
- * ```
- *
- * If you attempt to set the type on a 304 or 204 status code response, the
- * content type will not take effect as these status codes do not have content-types.
- *
- * ### Returning the current content type
- *
- * ```
- * type();
- * ```
- *
- * ### Storing content type definitions
- *
- * ```
- * type(['keynote' => 'application/keynote', 'bat' => 'application/bat']);
- * ```
- *
- * ### Replacing a content type definition
- *
- * ```
- * type(['jpg' => 'text/plain']);
- * ```
- *
- * @param string|null $contentType Content type key.
- * @return mixed Current content type or false if supplied an invalid content type.
- * @deprecated 3.5.5 Use getType() or withType() instead.
- */
- public function type($contentType = null)
- {
- deprecationWarning(
- 'Response::type() is deprecated. ' .
- 'Use getType() or withType() instead.'
- );
- if ($contentType === null) {
- return $this->getType();
- }
- if (is_array($contentType)) {
- foreach ($contentType as $type => $definition) {
- $this->_mimeTypes[$type] = $definition;
- }
- return $this->getType();
- }
- if (isset($this->_mimeTypes[$contentType])) {
- $contentType = $this->_mimeTypes[$contentType];
- $contentType = is_array($contentType) ? current($contentType) : $contentType;
- }
- if (strpos($contentType, '/') === false) {
- return false;
- }
- $this->_contentType = $contentType;
- $this->_setContentType();
- return $contentType;
- }
- /**
- * Returns the current content type.
- *
- * @return string
- */
- public function getType()
- {
- return $this->_contentType;
- }
- /**
- * Get an updated response with the content type set.
- *
- * If you attempt to set the type on a 304 or 204 status code response, the
- * content type will not take effect as these status codes do not have content-types.
- *
- * @param string $contentType Either a file extension which will be mapped to a mime-type or a concrete mime-type.
- * @return static
- */
- public function withType($contentType)
- {
- $mappedType = $this->resolveType($contentType);
- $new = clone $this;
- $new->_contentType = $mappedType;
- $new->_setContentType();
- return $new;
- }
- /**
- * Translate and validate content-types.
- *
- * @param string $contentType The content-type or type alias.
- * @return string The resolved content-type
- * @throws \InvalidArgumentException When an invalid content-type or alias is used.
- */
- protected function resolveType($contentType)
- {
- $mapped = $this->getMimeType($contentType);
- if ($mapped) {
- return is_array($mapped) ? current($mapped) : $mapped;
- }
- if (strpos($contentType, '/') === false) {
- throw new InvalidArgumentException(sprintf('"%s" is an invalid content type.', $contentType));
- }
- return $contentType;
- }
- /**
- * Returns the mime type definition for an alias
- *
- * e.g `getMimeType('pdf'); // returns 'application/pdf'`
- *
- * @param string $alias the content type alias to map
- * @return mixed String mapped mime type or false if $alias is not mapped
- */
- public function getMimeType($alias)
- {
- if (isset($this->_mimeTypes[$alias])) {
- return $this->_mimeTypes[$alias];
- }
- return false;
- }
- /**
- * Maps a content-type back to an alias
- *
- * e.g `mapType('application/pdf'); // returns 'pdf'`
- *
- * @param string|array $ctype Either a string content type to map, or an array of types.
- * @return string|array|null Aliases for the types provided.
- */
- public function mapType($ctype)
- {
- if (is_array($ctype)) {
- return array_map([$this, 'mapType'], $ctype);
- }
- foreach ($this->_mimeTypes as $alias => $types) {
- if (in_array($ctype, (array)$types)) {
- return $alias;
- }
- }
- return null;
- }
- /**
- * Sets the response charset
- * if $charset is null the current charset is returned
- *
- * @param string|null $charset Character set string.
- * @return string Current charset
- * @deprecated 3.5.0 Use getCharset()/withCharset() instead.
- */
- public function charset($charset = null)
- {
- deprecationWarning(
- 'Response::charset() is deprecated. ' .
- 'Use getCharset()/withCharset() instead.'
- );
- if ($charset === null) {
- return $this->_charset;
- }
- $this->_charset = $charset;
- $this->_setContentType();
- return $this->_charset;
- }
- /**
- * Returns the current charset.
- *
- * @return string
- */
- public function getCharset()
- {
- return $this->_charset;
- }
- /**
- * Get a new instance with an updated charset.
- *
- * @param string $charset Character set string.
- * @return static
- */
- public function withCharset($charset)
- {
- $new = clone $this;
- $new->_charset = $charset;
- $new->_setContentType();
- return $new;
- }
- /**
- * Sets the correct headers to instruct the client to not cache the response
- *
- * @return void
- * @deprecated 3.4.0 Use withDisabledCache() instead.
- */
- public function disableCache()
- {
- deprecationWarning(
- 'Response::disableCache() is deprecated. ' .
- 'Use withDisabledCache() instead.'
- );
- $this->_setHeader('Expires', 'Mon, 26 Jul 1997 05:00:00 GMT');
- $this->_setHeader('Last-Modified', gmdate('D, d M Y H:i:s') . ' GMT');
- $this->_setHeader('Cache-Control', 'no-store, no-cache, must-revalidate, post-check=0, pre-check=0');
- }
- /**
- * Create a new instance with headers to instruct the client to not cache the response
- *
- * @return static
- */
- public function withDisabledCache()
- {
- return $this->withHeader('Expires', 'Mon, 26 Jul 1997 05:00:00 GMT')
- ->withHeader('Last-Modified', gmdate('D, d M Y H:i:s') . ' GMT')
- ->withHeader('Cache-Control', 'no-store, no-cache, must-revalidate, post-check=0, pre-check=0');
- }
- /**
- * Sets the correct headers to instruct the client to cache the response.
- *
- * @param string $since a valid time since the response text has not been modified
- * @param string $time a valid time for cache expiry
- * @return void
- * @deprecated 3.4.0 Use withCache() instead.
- */
- public function cache($since, $time = '+1 day')
- {
- deprecationWarning(
- 'Response::cache() is deprecated. ' .
- 'Use withCache() instead.'
- );
- if (!is_int($time)) {
- $time = strtotime($time);
- }
- $this->_setHeader('Date', gmdate('D, j M Y G:i:s ', time()) . 'GMT');
- $this->modified($since);
- $this->expires($time);
- $this->sharable(true);
- $this->maxAge($time - time());
- }
- /**
- * Create a new instance with the headers to enable client caching.
- *
- * @param string $since a valid time since the response text has not been modified
- * @param string $time a valid time for cache expiry
- * @return static
- */
- public function withCache($since, $time = '+1 day')
- {
- if (!is_int($time)) {
- $time = strtotime($time);
- }
- return $this->withHeader('Date', gmdate('D, j M Y G:i:s ', time()) . 'GMT')
- ->withModified($since)
- ->withExpires($time)
- ->withSharable(true)
- ->withMaxAge($time - time());
- }
- /**
- * Sets whether a response is eligible to be cached by intermediate proxies
- * This method controls the `public` or `private` directive in the Cache-Control
- * header
- *
- * @param bool|null $public If set to true, the Cache-Control header will be set as public
- * if set to false, the response will be set to private
- * if no value is provided, it will return whether the response is sharable or not
- * @param int|null $time time in seconds after which the response should no longer be considered fresh
- * @return bool|null
- */
- public function sharable($public = null, $time = null)
- {
- if ($public === null) {
- $public = array_key_exists('public', $this->_cacheDirectives);
- $private = array_key_exists('private', $this->_cacheDirectives);
- $noCache = array_key_exists('no-cache', $this->_cacheDirectives);
- if (!$public && !$private && !$noCache) {
- return null;
- }
- return $public || !($private || $noCache);
- }
- if ($public) {
- $this->_cacheDirectives['public'] = true;
- unset($this->_cacheDirectives['private']);
- } else {
- $this->_cacheDirectives['private'] = true;
- unset($this->_cacheDirectives['public']);
- }
- $this->maxAge($time);
- if (!$time) {
- $this->_setCacheControl();
- }
- return (bool)$public;
- }
- /**
- * Create a new instace with the public/private Cache-Control directive set.
- *
- * @param bool $public If set to true, the Cache-Control header will be set as public
- * if set to false, the response will be set to private.
- * @param int|null $time time in seconds after which the response should no longer be considered fresh.
- * @return static
- */
- public function withSharable($public, $time = null)
- {
- $new = clone $this;
- unset($new->_cacheDirectives['private'], $new->_cacheDirectives['public']);
- $key = $public ? 'public' : 'private';
- $new->_cacheDirectives[$key] = true;
- if ($time !== null) {
- $new->_cacheDirectives['max-age'] = $time;
- }
- $new->_setCacheControl();
- return $new;
- }
- /**
- * Sets the Cache-Control s-maxage directive.
- *
- * The max-age is the number of seconds after which the response should no longer be considered
- * a good candidate to be fetched from a shared cache (like in a proxy server).
- * If called with no parameters, this function will return the current max-age value if any
- *
- * @param int|null $seconds if null, the method will return the current s-maxage value
- * @return int|null
- */
- public function sharedMaxAge($seconds = null)
- {
- if ($seconds !== null) {
- $this->_cacheDirectives['s-maxage'] = $seconds;
- $this->_setCacheControl();
- }
- if (isset($this->_cacheDirectives['s-maxage'])) {
- return $this->_cacheDirectives['s-maxage'];
- }
- return null;
- }
- /**
- * Create a new instance with the Cache-Control s-maxage directive.
- *
- * The max-age is the number of seconds after which the response should no longer be considered
- * a good candidate to be fetched from a shared cache (like in a proxy server).
- *
- * @param int $seconds The number of seconds for shared max-age
- * @return static
- */
- public function withSharedMaxAge($seconds)
- {
- $new = clone $this;
- $new->_cacheDirectives['s-maxage'] = $seconds;
- $new->_setCacheControl();
- return $new;
- }
- /**
- * Sets the Cache-Control max-age directive.
- * The max-age is the number of seconds after which the response should no longer be considered
- * a good candidate to be fetched from the local (client) cache.
- * If called with no parameters, this function will return the current max-age value if any
- *
- * @param int|null $seconds if null, the method will return the current max-age value
- * @return int|null
- */
- public function maxAge($seconds = null)
- {
- if ($seconds !== null) {
- $this->_cacheDirectives['max-age'] = $seconds;
- $this->_setCacheControl();
- }
- if (isset($this->_cacheDirectives['max-age'])) {
- return $this->_cacheDirectives['max-age'];
- }
- return null;
- }
- /**
- * Create an instance with Cache-Control max-age directive set.
- *
- * The max-age is the number of seconds after which the response should no longer be considered
- * a good candidate to be fetched from the local (client) cache.
- *
- * @param int $seconds The seconds a cached response can be considered valid
- * @return static
- */
- public function withMaxAge($seconds)
- {
- $new = clone $this;
- $new->_cacheDirectives['max-age'] = $seconds;
- $new->_setCacheControl();
- return $new;
- }
- /**
- * Sets the Cache-Control must-revalidate directive.
- * must-revalidate indicates that the response should not be served
- * stale by a cache under any circumstance without first revalidating
- * with the origin.
- * If called with no parameters, this function will return whether must-revalidate is present.
- *
- * @param bool|null $enable if null, the method will return the current
- * must-revalidate value. If boolean sets or unsets the directive.
- * @return bool
- * @deprecated 3.4.0 Use withMustRevalidate() instead.
- */
- public function mustRevalidate($enable = null)
- {
- deprecationWarning(
- 'Response::mustRevalidate() is deprecated. ' .
- 'Use withMustRevalidate() instead.'
- );
- if ($enable !== null) {
- if ($enable) {
- $this->_cacheDirectives['must-revalidate'] = true;
- } else {
- unset($this->_cacheDirectives['must-revalidate']);
- }
- $this->_setCacheControl();
- }
- return array_key_exists('must-revalidate', $this->_cacheDirectives);
- }
- /**
- * Create an instance with Cache-Control must-revalidate directive set.
- *
- * Sets the Cache-Control must-revalidate directive.
- * must-revalidate indicates that the response should not be served
- * stale by a cache under any circumstance without first revalidating
- * with the origin.
- *
- * @param bool $enable If boolean sets or unsets the directive.
- * @return static
- */
- public function withMustRevalidate($enable)
- {
- $new = clone $this;
- if ($enable) {
- $new->_cacheDirectives['must-revalidate'] = true;
- } else {
- unset($new->_cacheDirectives['must-revalidate']);
- }
- $new->_setCacheControl();
- return $new;
- }
- /**
- * Helper method to generate a valid Cache-Control header from the options set
- * in other methods
- *
- * @return void
- */
- protected function _setCacheControl()
- {
- $control = '';
- foreach ($this->_cacheDirectives as $key => $val) {
- $control .= $val === true ? $key : sprintf('%s=%s', $key, $val);
- $control .= ', ';
- }
- $control = rtrim($control, ', ');
- $this->_setHeader('Cache-Control', $control);
- }
- /**
- * Sets the Expires header for the response by taking an expiration time
- * If called with no parameters it will return the current Expires value
- *
- * ### Examples:
- *
- * `$response->expires('now')` Will Expire the response cache now
- * `$response->expires(new DateTime('+1 day'))` Will set the expiration in next 24 hours
- * `$response->expires()` Will return the current expiration header value
- *
- * @param string|\DateTime|null $time Valid time string or \DateTime instance.
- * @return string|null
- * @deprecated 3.4.0 Use withExpires() instead.
- */
- public function expires($time = null)
- {
- deprecationWarning(
- 'Response::expires() is deprecated. ' .
- 'Use withExpires() instead.'
- );
- if ($time !== null) {
- $date = $this->_getUTCDate($time);
- $this->_setHeader('Expires', $date->format('D, j M Y H:i:s') . ' GMT');
- }
- if ($this->hasHeader('Expires')) {
- return $this->getHeaderLine('Expires');
- }
- return null;
- }
- /**
- * Create a new instance with the Expires header set.
- *
- * ### Examples:
- *
- * ```
- * // Will Expire the response cache now
- * $response->withExpires('now')
- *
- * // Will set the expiration in next 24 hours
- * $response->withExpires(new DateTime('+1 day'))
- * ```
- *
- * @param string|\DateTime $time Valid time string or \DateTime instance.
- * @return static
- */
- public function withExpires($time)
- {
- $date = $this->_getUTCDate($time);
- return $this->withHeader('Expires', $date->format('D, j M Y H:i:s') . ' GMT');
- }
- /**
- * Sets the Last-Modified header for the response by taking a modification time
- * If called with no parameters it will return the current Last-Modified value
- *
- * ### Examples:
- *
- * `$response->modified('now')` Will set the Last-Modified to the current time
- * `$response->modified(new DateTime('+1 day'))` Will set the modification date in the past 24 hours
- * `$response->modified()` Will return the current Last-Modified header value
- *
- * @param string|\DateTime|null $time Valid time string or \DateTime instance.
- * @return string|null
- * @deprecated 3.4.0 Use withModified() instead.
- */
- public function modified($time = null)
- {
- deprecationWarning(
- 'Response::modified() is deprecated. ' .
- 'Use withModified() or getHeaderLine("Last-Modified") instead.'
- );
- if ($time !== null) {
- $date = $this->_getUTCDate($time);
- $this->_setHeader('Last-Modified', $date->format('D, j M Y H:i:s') . ' GMT');
- }
- if ($this->hasHeader('Last-Modified')) {
- return $this->getHeaderLine('Last-Modified');
- }
- return null;
- }
- /**
- * Create a new instance with the Last-Modified header set.
- *
- * ### Examples:
- *
- * ```
- * // Will Expire the response cache now
- * $response->withModified('now')
- *
- * // Will set the expiration in next 24 hours
- * $response->withModified(new DateTime('+1 day'))
- * ```
- *
- * @param string|\DateTime $time Valid time string or \DateTime instance.
- * @return static
- */
- public function withModified($time)
- {
- $date = $this->_getUTCDate($time);
- return $this->withHeader('Last-Modified', $date->format('D, j M Y H:i:s') . ' GMT');
- }
- /**
- * Sets the response as Not Modified by removing any body contents
- * setting the status code to "304 Not Modified" and removing all
- * conflicting headers
- *
- * *Warning* This method mutates the response in-place and should be avoided.
- *
- * @return void
- */
- public function notModified()
- {
- $this->_createStream();
- $this->_setStatus(304);
- $remove = [
- 'Allow',
- 'Content-Encoding',
- 'Content-Language',
- 'Content-Length',
- 'Content-MD5',
- 'Content-Type',
- 'Last-Modified'
- ];
- foreach ($remove as $header) {
- $this->_clearHeader($header);
- }
- }
- /**
- * Create a new instance as 'not modified'
- *
- * This will remove any body contents set the status code
- * to "304" and removing headers that describe
- * a response body.
- *
- * @return static
- */
- public function withNotModified()
- {
- $new = $this->withStatus(304);
- $new->_createStream();
- $remove = [
- 'Allow',
- 'Content-Encoding',
- 'Content-Language',
- 'Content-Length',
- 'Content-MD5',
- 'Content-Type',
- 'Last-Modified'
- ];
- foreach ($remove as $header) {
- $new = $new->withoutHeader($header);
- }
- return $new;
- }
- /**
- * Sets the Vary header for the response, if an array is passed,
- * values will be imploded into a comma separated string. If no
- * parameters are passed, then an array with the current Vary header
- * value is returned
- *
- * @param string|array|null $cacheVariances A single Vary string or an array
- * containing the list for variances.
- * @return array|null
- * @deprecated 3.4.0 Use withVary() instead.
- */
- public function vary($cacheVariances = null)
- {
- deprecationWarning(
- 'Response::vary() is deprecated. ' .
- 'Use withVary() instead.'
- );
- if ($cacheVariances !== null) {
- $cacheVariances = (array)$cacheVariances;
- $this->_setHeader('Vary', implode(', ', $cacheVariances));
- }
- if ($this->hasHeader('Vary')) {
- return explode(', ', $this->getHeaderLine('Vary'));
- }
- return null;
- }
- /**
- * Create a new instance with the Vary header set.
- *
- * If an array is passed values will be imploded into a comma
- * separated string. If no parameters are passed, then an
- * array with the current Vary header value is returned
- *
- * @param string|array $cacheVariances A single Vary string or an array
- * containing the list for variances.
- * @return static
- */
- public function withVary($cacheVariances)
- {
- return $this->withHeader('Vary', (array)$cacheVariances);
- }
- /**
- * Sets the response Etag, Etags are a strong indicative that a response
- * can be cached by a HTTP client. A bad way of generating Etags is
- * creating a hash of the response output, instead generate a unique
- * hash of the unique components that identifies a request, such as a
- * modification time, a resource Id, and anything else you consider it
- * makes it unique.
- *
- * Second parameter is used to instruct clients that the content has
- * changed, but semantically, it can be used as the same thing. Think
- * for instance of a page with a hit counter, two different page views
- * are equivalent, but they differ by a few bytes. This leaves off to
- * the Client the decision of using or not the cached page.
- *
- * If no parameters are passed, current Etag header is returned.
- *
- * @param string|null $hash The unique hash that identifies this response
- * @param bool $weak Whether the response is semantically the same as
- * other with the same hash or not
- * @return string|null
- * @deprecated 3.4.0 Use withEtag() instead.
- */
- public function etag($hash = null, $weak = false)
- {
- deprecationWarning(
- 'Response::etag() is deprecated. ' .
- 'Use withEtag() or getHeaderLine("Etag") instead.'
- );
- if ($hash !== null) {
- $this->_setHeader('Etag', sprintf('%s"%s"', $weak ? 'W/' : null, $hash));
- }
- if ($this->hasHeader('Etag')) {
- return $this->getHeaderLine('Etag');
- }
- return null;
- }
- /**
- * Create a new instance with the Etag header set.
- *
- * Etags are a strong indicative that a response can be cached by a
- * HTTP client. A bad way of generating Etags is creating a hash of
- * the response output, instead generate a unique hash of the
- * unique components that identifies a request, such as a
- * modification time, a resource Id, and anything else you consider it
- * that makes the response unique.
- *
- * The second parameter is used to inform clients that the content has
- * changed, but semantically it is equivalent to existing cached values. Consider
- * a page with a hit counter, two different page views are equivalent, but
- * they differ by a few bytes. This permits the Client to decide whether they should
- * use the cached data.
- *
- * @param string $hash The unique hash that identifies this response
- * @param bool $weak Whether the response is semantically the same as
- * other with the same hash or not. Defaults to false
- * @return static
- */
- public function withEtag($hash, $weak = false)
- {
- $hash = sprintf('%s"%s"', $weak ? 'W/' : null, $hash);
- return $this->withHeader('Etag', $hash);
- }
- /**
- * Returns a DateTime object initialized at the $time param and using UTC
- * as timezone
- *
- * @param string|int|\DateTime|null $time Valid time string or \DateTime instance.
- * @return \DateTime
- */
- protected function _getUTCDate($time = null)
- {
- if ($time instanceof DateTime) {
- $result = clone $time;
- } elseif (is_int($time)) {
- $result = new DateTime(date('Y-m-d H:i:s', $time));
- } else {
- $result = new DateTime($time);
- }
- $result->setTimezone(new DateTimeZone('UTC'));
- return $result;
- }
- /**
- * Sets the correct output buffering handler to send a compressed response. Responses will
- * be compressed with zlib, if the extension is available.
- *
- * @return bool false if client does not accept compressed responses or no handler is available, true otherwise
- */
- public function compress()
- {
- $compressionEnabled = ini_get('zlib.output_compression') !== '1' &&
- extension_loaded('zlib') &&
- (strpos(env('HTTP_ACCEPT_ENCODING'), 'gzip') !== false);
- return $compressionEnabled && ob_start('ob_gzhandler');
- }
- /**
- * Returns whether the resulting output will be compressed by PHP
- *
- * @return bool
- */
- public function outputCompressed()
- {
- return strpos(env('HTTP_ACCEPT_ENCODING'), 'gzip') !== false
- && (ini_get('zlib.output_compression') === '1' || in_array('ob_gzhandler', ob_list_handlers()));
- }
- /**
- * Sets the correct headers to instruct the browser to download the response as a file.
- *
- * @param string $filename The name of the file as the browser will download the response
- * @return void
- * @deprecated 3.4.0 Use withDownload() instead.
- */
- public function download($filename)
- {
- deprecationWarning(
- 'Response::download() is deprecated. ' .
- 'Use withDownload() instead.'
- );
- $this->header('Content-Disposition', 'attachment; filename="' . $filename . '"');
- }
- /**
- * Create a new instance with the Content-Disposition header set.
- *
- * @param string $filename The name of the file as the browser will download the response
- * @return static
- */
- public function withDownload($filename)
- {
- return $this->withHeader('Content-Disposition', 'attachment; filename="' . $filename . '"');
- }
- /**
- * Sets the protocol to be used when sending the response. Defaults to HTTP/1.1
- * If called with no arguments, it will return the current configured protocol
- *
- * @param string|null $protocol Protocol to be used for sending response.
- * @return string Protocol currently set
- * @deprecated 3.4.0 Use getProtocolVersion() instead.
- */
- public function protocol($protocol = null)
- {
- deprecationWarning(
- 'Response::protocol() is deprecated. ' .
- 'Use getProtocolVersion() instead.'
- );
- if ($protocol !== null) {
- $this->_protocol = $protocol;
- }
- return $this->_protocol;
- }
- /**
- * Sets the Content-Length header for the response
- * If called with no arguments returns the last Content-Length set
- *
- * @param int|null $bytes Number of bytes
- * @return string|null
- * @deprecated 3.4.0 Use withLength() to set length instead.
- */
- public function length($bytes = null)
- {
- deprecationWarning(
- 'Response::length() is deprecated. ' .
- 'Use withLength() instead.'
- );
- if ($bytes !== null) {
- $this->_setHeader('Content-Length', $bytes);
- }
- if ($this->hasHeader('Content-Length')) {
- return $this->getHeaderLine('Content-Length');
- }
- return null;
- }
- /**
- * Create a new response with the Content-Length header set.
- *
- * @param int|string $bytes Number of bytes
- * @return static
- */
- public function withLength($bytes)
- {
- return $this->withHeader('Content-Length', (string)$bytes);
- }
- /**
- * Checks whether a response has not been modified according to the 'If-None-Match'
- * (Etags) and 'If-Modified-Since' (last modification date) request
- * headers. If the response is detected to be not modified, it
- * is marked as so accordingly so the client can be informed of that.
- *
- * In order to mark a response as not modified, you need to set at least
- * the Last-Modified etag response header before calling this method. Otherwise
- * a comparison will not be possible.
- *
- * *Warning* This method mutates the response in-place and should be avoided.
- *
- * @param \Cake\Http\ServerRequest $request Request object
- * @return bool Whether the response was marked as not modified or not.
- */
- public function checkNotModified(ServerRequest $request)
- {
- $etags = preg_split('/\s*,\s*/', (string)$request->getHeaderLine('If-None-Match'), 0, PREG_SPLIT_NO_EMPTY);
- $responseTag = $this->getHeaderLine('Etag');
- if ($responseTag) {
- $etagMatches = in_array('*', $etags) || in_array($responseTag, $etags);
- }
- $modifiedSince = $request->getHeaderLine('If-Modified-Since');
- if ($modifiedSince && $this->hasHeader('Last-Modified')) {
- $timeMatches = strtotime($this->getHeaderLine('Last-Modified')) === strtotime($modifiedSince);
- }
- $checks = compact('etagMatches', 'timeMatches');
- if (empty($checks)) {
- return false;
- }
- $notModified = !in_array(false, $checks, true);
- if ($notModified) {
- $this->notModified();
- }
- return $notModified;
- }
- /**
- * String conversion. Fetches the response body as a string.
- * Does *not* send headers.
- * If body is a callable, a blank string is returned.
- *
- * @return string
- */
- public function __toString()
- {
- $this->stream->rewind();
- return (string)$this->stream->getContents();
- }
- /**
- * Getter/Setter for cookie configs
- *
- * This method acts as a setter/getter depending on the type of the argument.
- * If the method is called with no arguments, it returns all configurations.
- *
- * If the method is called with a string as argument, it returns either the
- * given configuration if it is set, or null, if it's not set.
- *
- * If the method is called with an array as argument, it will set the cookie
- * configuration to the cookie container.
- *
- * ### Options (when setting a configuration)
- * - name: The Cookie name
- * - value: Value of the cookie
- * - expire: Time the cookie expires in
- * - path: Path the cookie applies to
- * - domain: Domain the cookie is for.
- * - secure: Is the cookie https?
- * - httpOnly: Is the cookie available in the client?
- *
- * ### Examples
- *
- * ### Getting all cookies
- *
- * `$this->cookie()`
- *
- * ### Getting a certain cookie configuration
- *
- * `$this->cookie('MyCookie')`
- *
- * ### Setting a cookie configuration
- *
- * `$this->cookie((array) $options)`
- *
- * @param array|null $options Either null to get all cookies, string for a specific cookie
- * or array to set cookie.
- * @return mixed
- * @deprecated 3.4.0 Use getCookie(), getCookies() and withCookie() instead.
- */
- public function cookie($options = null)
- {
- deprecationWarning(
- 'Response::cookie() is deprecated. ' .
- 'Use getCookie(), getCookies() and withCookie() instead.'
- );
- if ($options === null) {
- return $this->getCookies();
- }
- if (is_string($options)) {
- if (!$this->_cookies->has($options)) {
- return null;
- }
- $cookie = $this->_cookies->get($options);
- return $this->convertCookieToArray($cookie);
- }
- $options += [
- 'name' => 'CakeCookie[default]',
- 'value' => '',
- 'expire' => 0,
- 'path' => '/',
- 'domain' => '',
- 'secure' => false,
- 'httpOnly' => false
- ];
- $expires = $options['expire'] ? new DateTime('@' . $options['expire']) : null;
- $cookie = new Cookie(
- $options['name'],
- $options['value'],
- $expires,
- $options['path'],
- $options['domain'],
- $options['secure'],
- $options['httpOnly']
- );
- $this->_cookies = $this->_cookies->add($cookie);
- }
- /**
- * Create a new response with a cookie set.
- *
- * ### Options
- *
- * - `name`: The Cookie name
- * - `value`: Value of the cookie
- * - `expire`: Time the cookie expires in
- * - `path`: Path the cookie applies to
- * - `domain`: Domain the cookie is for.
- * - `secure`: Is the cookie https?
- * - `httpOnly`: Is the cookie available in the client?
- *
- * ### Examples
- *
- * ```
- * // set scalar value with defaults
- * $response = $response->withCookie('remember_me', 1);
- *
- * // customize cookie attributes
- * $response = $response->withCookie('remember_me', ['path' => '/login']);
- *
- * // add a cookie object
- * $response = $response->withCookie(new Cookie('remember_me', 1));
- * ```
- *
- * @param string|\Cake\Http\Cookie\Cookie $name The name of the cookie to set, or a cookie object
- * @param array|string $data Either a string value, or an array of cookie options.
- * @return static
- */
- public function withCookie($name, $data = '')
- {
- if ($name instanceof Cookie) {
- $cookie = $name;
- } else {
- if (!is_array($data)) {
- $data = ['value' => $data];
- }
- $data += [
- 'value' => '',
- 'expire' => 0,
- 'path' => '/',
- 'domain' => '',
- 'secure' => false,
- 'httpOnly' => false
- ];
- $expires = $data['expire'] ? new DateTime('@' . $data['expire']) : null;
- $cookie = new Cookie(
- $name,
- $data['value'],
- $expires,
- $data['path'],
- $data['domain'],
- $data['secure'],
- $data['httpOnly']
- );
- }
- $new = clone $this;
- $new->_cookies = $new->_cookies->add($cookie);
- return $new;
- }
- /**
- * Create a new response with an expired cookie set.
- *
- * ### Options
- *
- * - `path`: Path the cookie applies to
- * - `domain`: Domain the cookie is for.
- * - `secure`: Is the cookie https?
- * - `httpOnly`: Is the cookie available in the client?
- *
- * ### Examples
- *
- * ```
- * // set scalar value with defaults
- * $response = $response->withExpiredCookie('remember_me');
- *
- * // customize cookie attributes
- * $response = $response->withExpiredCookie('remember_me', ['path' => '/login']);
- *
- * // add a cookie object
- * $response = $response->withExpiredCookie(new Cookie('remember_me'));
- * ```
- *
- * @param string|\Cake\Http\Cookie\CookieInterface $name The name of the cookie to expire, or a cookie object
- * @param array $options An array of cookie options.
- * @return static
- */
- public function withExpiredCookie($name, $options = [])
- {
- if ($name instanceof CookieInterface) {
- $cookie = $name->withExpired();
- } else {
- $options += [
- 'path' => '/',
- 'domain' => '',
- 'secure' => false,
- 'httpOnly' => false
- ];
- $cookie = new Cookie(
- $name,
- '',
- DateTime::createFromFormat('U', 1),
- $options['path'],
- $options['domain'],
- $options['secure'],
- $options['httpOnly']
- );
- }
- $new = clone $this;
- $new->_cookies = $new->_cookies->add($cookie);
- return $new;
- }
- /**
- * Read a single cookie from the response.
- *
- * This method provides read access to pending cookies. It will
- * not read the `Set-Cookie` header if set.
- *
- * @param string $name The cookie name you want to read.
- * @return array|null Either the cookie data or null
- */
- public function getCookie($name)
- {
- if (!$this->_cookies->has($name)) {
- return null;
- }
- $cookie = $this->_cookies->get($name);
- return $this->convertCookieToArray($cookie);
- }
- /**
- * Get all cookies in the response.
- *
- * Returns an associative array of cookie name => cookie data.
- *
- * @return array
- */
- public function getCookies()
- {
- $out = [];
- foreach ($this->_cookies as $cookie) {
- $out[$cookie->getName()] = $this->convertCookieToArray($cookie);
- }
- return $out;
- }
- /**
- * Convert the cookie into an array of its properties.
- *
- * This method is compatible with the historical behavior of Cake\Http\Response,
- * where `httponly` is `httpOnly` and `expires` is `expire`
- *
- * @param \Cake\Http\Cookie\CookieInterface $cookie Cookie object.
- * @return array
- */
- protected function convertCookieToArray(CookieInterface $cookie)
- {
- return [
- 'name' => $cookie->getName(),
- 'value' => $cookie->getStringValue(),
- 'path' => $cookie->getPath(),
- 'domain' => $cookie->getDomain(),
- 'secure' => $cookie->isSecure(),
- 'httpOnly' => $cookie->isHttpOnly(),
- 'expire' => $cookie->getExpiresTimestamp()
- ];
- }
- /**
- * Get the CookieCollection from the response
- *
- * @return \Cake\Http\Cookie\CookieCollection
- */
- public function getCookieCollection()
- {
- return $this->_cookies;
- }
- /**
- * Setup access for origin and methods on cross origin requests
- *
- * This method allow multiple ways to setup the domains, see the examples
- *
- * ### Full URI
- * ```
- * cors($request, 'https://www.cakephp.org');
- * ```
- *
- * ### URI with wildcard
- * ```
- * cors($request, 'https://*.cakephp.org');
- * ```
- *
- * ### Ignoring the requested protocol
- * ```
- * cors($request, 'www.cakephp.org');
- * ```
- *
- * ### Any URI
- * ```
- * cors($request, '*');
- * ```
- *
- * ### Whitelist of URIs
- * ```
- * cors($request, ['http://www.cakephp.org', '*.google.com', 'https://myproject.github.io']);
- * ```
- *
- * *Note* The `$allowedDomains`, `$allowedMethods`, `$allowedHeaders` parameters are deprecated.
- * Instead the builder object should be used.
- *
- * @param \Cake\Http\ServerRequest $request Request object
- * @param string|array $allowedDomains List of allowed domains, see method description for more details
- * @param string|array $allowedMethods List of HTTP verbs allowed
- * @param string|array $allowedHeaders List of HTTP headers allowed
- * @return \Cake\Http\CorsBuilder A builder object the provides a fluent interface for defining
- * additional CORS headers.
- */
- public function cors(ServerRequest $request, $allowedDomains = [], $allowedMethods = [], $allowedHeaders = [])
- {
- $origin = $request->getHeaderLine('Origin');
- $ssl = $request->is('ssl');
- $builder = new CorsBuilder($this, $origin, $ssl);
- if (!$origin) {
- return $builder;
- }
- if (empty($allowedDomains) && empty($allowedMethods) && empty($allowedHeaders)) {
- return $builder;
- }
- deprecationWarning(
- 'The $allowedDomains, $allowedMethods, and $allowedHeaders parameters of Response::cors() ' .
- 'are deprecated. Instead you should use the builder methods on the return of cors().'
- );
- $updated = $builder->allowOrigin($allowedDomains)
- ->allowMethods((array)$allowedMethods)
- ->allowHeaders((array)$allowedHeaders)
- ->build();
- // If $updated is a new instance, mutate this object in-place
- // to retain existing behavior.
- if ($updated !== $this) {
- foreach ($updated->getHeaders() as $name => $values) {
- if (!$this->hasHeader($name)) {
- $this->_setHeader($name, $values[0]);
- }
- }
- }
- return $builder;
- }
- /**
- * Setup for display or download the given file.
- *
- * If $_SERVER['HTTP_RANGE'] is set a slice of the file will be
- * returned instead of the entire file.
- *
- * ### Options keys
- *
- * - name: Alternate download name
- * - download: If `true` sets download header and forces file to be downloaded rather than displayed in browser
- *
- * @param string $path Path to file. If the path is not an absolute path that resolves
- * to a file, `APP` will be prepended to the path.
- * @param array $options Options See above.
- * @return void
- * @throws \Cake\Http\Exception\NotFoundException
- * @deprecated 3.4.0 Use withFile() instead.
- */
- public function file($path, array $options = [])
- {
- deprecationWarning(
- 'Response::file() is deprecated. ' .
- 'Use withFile() instead.'
- );
- $file = $this->validateFile($path);
- $options += [
- 'name' => null,
- 'download' => null
- ];
- $extension = strtolower($file->ext());
- $download = $options['download'];
- if ((!$extension || $this->type($extension) === false) && $download === null) {
- $download = true;
- }
- $fileSize = $file->size();
- if ($download) {
- $agent = env('HTTP_USER_AGENT');
- if (preg_match('%Opera(/| )([0-9].[0-9]{1,2})%', $agent)) {
- $contentType = 'application/octet-stream';
- } elseif (preg_match('/MSIE ([0-9].[0-9]{1,2})/', $agent)) {
- $contentType = 'application/force-download';
- }
- if (!empty($contentType)) {
- $this->type($contentType);
- }
- if ($options['name'] === null) {
- $name = $file->name;
- } else {
- $name = $options['name'];
- }
- $this->download($name);
- $this->header('Content-Transfer-Encoding', 'binary');
- }
- $this->header('Accept-Ranges', 'bytes');
- $httpRange = env('HTTP_RANGE');
- if (isset($httpRange)) {
- $this->_fileRange($file, $httpRange);
- } else {
- $this->header('Content-Length', $fileSize);
- }
- $this->_file = $file;
- $this->stream = new Stream($file->path, 'rb');
- }
- /**
- * Create a new instance that is based on a file.
- *
- * This method will augment both the body and a number of related headers.
- *
- * If `$_SERVER['HTTP_RANGE']` is set, a slice of the file will be
- * returned instead of the entire file.
- *
- * ### Options keys
- *
- * - name: Alternate download name
- * - download: If `true` sets download header and forces file to
- * be downloaded rather than displayed inline.
- *
- * @param string $path Path to file. If the path is not an absolute path that resolves
- * to a file, `APP` will be prepended to the path.
- * @param array $options Options See above.
- * @return static
- * @throws \Cake\Http\Exception\NotFoundException
- */
- public function withFile($path, array $options = [])
- {
- $file = $this->validateFile($path);
- $options += [
- 'name' => null,
- 'download' => null
- ];
- $extension = strtolower($file->ext());
- $mapped = $this->getMimeType($extension);
- if ((!$extension || !$mapped) && $options['download'] === null) {
- $options['download'] = true;
- }
- $new = clone $this;
- if ($mapped) {
- $new = $new->withType($extension);
- }
- $fileSize = $file->size();
- if ($options['download']) {
- $agent = env('HTTP_USER_AGENT');
- if (preg_match('%Opera(/| )([0-9].[0-9]{1,2})%', $agent)) {
- $contentType = 'application/octet-stream';
- } elseif (preg_match('/MSIE ([0-9].[0-9]{1,2})/', $agent)) {
- $contentType = 'application/force-download';
- }
- if (isset($contentType)) {
- $new = $new->withType($contentType);
- }
- $name = $options['name'] ?: $file->name;
- $new = $new->withDownload($name)
- ->withHeader('Content-Transfer-Encoding', 'binary');
- }
- $new = $new->withHeader('Accept-Ranges', 'bytes');
- $httpRange = env('HTTP_RANGE');
- if (isset($httpRange)) {
- $new->_fileRange($file, $httpRange);
- } else {
- $new = $new->withHeader('Content-Length', (string)$fileSize);
- }
- $new->_file = $file;
- $new->stream = new Stream($file->path, 'rb');
- return $new;
- }
- /**
- * Convenience method to set a string into the response body
- *
- * @param string $string The string to be sent
- * @return static
- */
- public function withStringBody($string)
- {
- $new = clone $this;
- $new->_createStream();
- $new->stream->write((string)$string);
- return $new;
- }
- /**
- * Validate a file path is a valid response body.
- *
- * @param string $path The path to the file.
- * @throws \Cake\Http\Exception\NotFoundException
- * @return \Cake\Filesystem\File
- */
- protected function validateFile($path)
- {
- if (strpos($path, '../') !== false || strpos($path, '..\\') !== false) {
- throw new NotFoundException(__d('cake', 'The requested file contains `..` and will not be read.'));
- }
- if (!is_file($path)) {
- $path = APP . $path;
- }
- $file = new File($path);
- if (!$file->exists() || !$file->readable()) {
- if (Configure::read('debug')) {
- throw new NotFoundException(sprintf('The requested file %s was not found or not readable', $path));
- }
- throw new NotFoundException(__d('cake', 'The requested file was not found'));
- }
- return $file;
- }
- /**
- * Get the current file if one exists.
- *
- * @return \Cake\Filesystem\File|null The file to use in the response or null
- */
- public function getFile()
- {
- return $this->_file;
- }
- /**
- * Apply a file range to a file and set the end offset.
- *
- * If an invalid range is requested a 416 Status code will be used
- * in the response.
- *
- * @param \Cake\Filesystem\File $file The file to set a range on.
- * @param string $httpRange The range to use.
- * @return void
- * @deprecated 3.4.0 Long term this needs to be refactored to follow immutable paradigms.
- * However for now, it is simpler to leave this alone.
- */
- protected function _fileRange($file, $httpRange)
- {
- $fileSize = $file->size();
- $lastByte = $fileSize - 1;
- $start = 0;
- $end = $lastByte;
- preg_match('/^bytes\s*=\s*(\d+)?\s*-\s*(\d+)?$/', $httpRange, $matches);
- if ($matches) {
- $start = $matches[1];
- $end = isset($matches[2]) ? $matches[2] : '';
- }
- if ($start === '') {
- $start = $fileSize - $end;
- $end = $lastByte;
- }
- if ($end === '') {
- $end = $lastByte;
- }
- if ($start > $end || $end > $lastByte || $start > $lastByte) {
- $this->_setStatus(416);
- $this->_setHeader('Content-Range', 'bytes 0-' . $lastByte . '/' . $fileSize);
- return;
- }
- $this->_setHeader('Content-Length', $end - $start + 1);
- $this->_setHeader('Content-Range', 'bytes ' . $start . '-' . $end . '/' . $fileSize);
- $this->_setStatus(206);
- $this->_fileRange = [$start, $end];
- }
- /**
- * Reads out a file, and echos the content to the client.
- *
- * @param \Cake\Filesystem\File $file File object
- * @param array $range The range to read out of the file.
- * @return bool True is whole file is echoed successfully or false if client connection is lost in between
- * @deprecated 3.4.0 Will be removed in 4.0.0
- */
- protected function _sendFile($file, $range)
- {
- deprecationWarning('Will be removed in 4.0.0');
- ob_implicit_flush(true);
- $file->open('rb');
- $end = $start = false;
- if ($range) {
- list($start, $end) = $range;
- }
- if ($start !== false) {
- $file->offset($start);
- }
- $bufferSize = 8192;
- set_time_limit(0);
- session_write_close();
- while (!feof($file->handle)) {
- if (!$this->_isActive()) {
- $file->close();
- return false;
- }
- $offset = $file->offset();
- if ($end && $offset >= $end) {
- break;
- }
- if ($end && $offset + $bufferSize >= $end) {
- $bufferSize = $end - $offset + 1;
- }
- echo fread($file->handle, $bufferSize);
- }
- $file->close();
- return true;
- }
- /**
- * Returns true if connection is still active
- *
- * @return bool
- * @deprecated 3.4.0 Will be removed in 4.0.0
- */
- protected function _isActive()
- {
- deprecationWarning('Will be removed in 4.0.0');
- return connection_status() === CONNECTION_NORMAL && !connection_aborted();
- }
- /**
- * Clears the contents of the topmost output buffer and discards them
- *
- * @return bool
- * @deprecated 3.2.4 This function is not needed anymore
- */
- protected function _clearBuffer()
- {
- deprecationWarning(
- 'This function is not needed anymore and will be removed.'
- );
- //@codingStandardsIgnoreStart
- return @ob_end_clean();
- //@codingStandardsIgnoreEnd
- }
- /**
- * Flushes the contents of the output buffer
- *
- * @return void
- * @deprecated 3.2.4 This function is not needed anymore
- */
- protected function _flushBuffer()
- {
- deprecationWarning(
- 'This function is not needed anymore and will be removed.'
- );
- //@codingStandardsIgnoreStart
- @flush();
- if (ob_get_level()) {
- @ob_flush();
- }
- //@codingStandardsIgnoreEnd
- }
- /**
- * Stop execution of the current script. Wraps exit() making
- * testing easier.
- *
- * @param int|string $status See https://secure.php.net/exit for values
- * @return void
- * @deprecated 3.4.0 Will be removed in 4.0.0
- */
- public function stop($status = 0)
- {
- deprecationWarning('Will be removed in 4.0.0');
- exit($status);
- }
- /**
- * Returns an array that can be used to describe the internal state of this
- * object.
- *
- * @return array
- */
- public function __debugInfo()
- {
- return [
- 'status' => $this->_status,
- 'contentType' => $this->_contentType,
- 'headers' => $this->headers,
- 'file' => $this->_file,
- 'fileRange' => $this->_fileRange,
- 'cookies' => $this->_cookies,
- 'cacheDirectives' => $this->_cacheDirectives,
- 'body' => (string)$this->getBody(),
- ];
- }
- }
- // @deprecated Add backwards compat alias.
- class_alias('Cake\Http\Response', 'Cake\Network\Response');
|