123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344 |
- <?php
- /**
- * CakeResponse
- *
- * PHP 5
- *
- * CakePHP(tm) : Rapid Development Framework (http://cakephp.org)
- * Copyright 2005-2012, Cake Software Foundation, Inc. (http://cakefoundation.org)
- *
- * Licensed under The MIT License
- * Redistributions of files must retain the above copyright notice.
- *
- * @copyright Copyright 2005-2012, Cake Software Foundation, Inc. (http://cakefoundation.org)
- * @link http://cakephp.org CakePHP(tm) Project
- * @package Cake.Network
- * @since CakePHP(tm) v 2.0
- * @license MIT License (http://www.opensource.org/licenses/mit-license.php)
- */
- App::uses('File', 'Utility');
- /**
- * CakeResponse is responsible for managing the response text, status and headers of a HTTP response.
- *
- * By default controllers will use this class to render their response. If you are going to use
- * a custom response class it should subclass this object in order to ensure compatibility.
- *
- * @package Cake.Network
- */
- class CakeResponse {
- /**
- * Holds HTTP response statuses
- *
- * @var array
- */
- protected $_statusCodes = array(
- 100 => 'Continue',
- 101 => 'Switching Protocols',
- 200 => 'OK',
- 201 => 'Created',
- 202 => 'Accepted',
- 203 => 'Non-Authoritative Information',
- 204 => 'No Content',
- 205 => 'Reset Content',
- 206 => 'Partial Content',
- 300 => 'Multiple Choices',
- 301 => 'Moved Permanently',
- 302 => 'Found',
- 303 => 'See Other',
- 304 => 'Not Modified',
- 305 => 'Use Proxy',
- 307 => 'Temporary 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 Time-out',
- 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',
- 500 => 'Internal Server Error',
- 501 => 'Not Implemented',
- 502 => 'Bad Gateway',
- 503 => 'Service Unavailable',
- 504 => 'Gateway Time-out'
- );
- /**
- * Holds known mime type mappings
- *
- * @var array
- */
- protected $_mimeTypes = array(
- 'html' => array('text/html', '*/*'),
- 'json' => 'application/json',
- 'xml' => array('application/xml', 'text/xml'),
- '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' => array('text/csv', 'application/vnd.ms-excel', 'text/plain'),
- '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',
- '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' => array('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',
- 'jpe' => 'image/jpeg',
- 'jpeg' => 'image/jpeg',
- 'jpg' => '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',
- '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' => array('application/xhtml+xml', 'application/xhtml', 'text/xhtml'),
- 'xhtml-mobile' => 'application/vnd.wap.xhtml+xml',
- 'atom' => 'application/atom+xml',
- 'amf' => 'application/x-amf',
- 'wap' => array('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',
- 'webp' => 'image/webp',
- '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',
- );
- /**
- * Protocol header to send to the client
- *
- * @var string
- */
- protected $_protocol = 'HTTP/1.1';
- /**
- * Status code to send to the client
- *
- * @var integer
- */
- 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 integer
- */
- protected $_contentType = 'text/html';
- /**
- * Buffer list of headers
- *
- * @var array
- */
- protected $_headers = array();
- /**
- * Buffer string for response message
- *
- * @var string
- */
- protected $_body = null;
- /**
- * File object for file to be read out as response
- *
- * @var File
- */
- protected $_file = null;
- /**
- * 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 string
- */
- protected $_cacheDirectives = array();
- /**
- * Holds cookies to be sent to the client
- *
- * @var array
- */
- protected $_cookies = array();
- /**
- * Class 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
- * - 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 = array()) {
- if (isset($options['body'])) {
- $this->body($options['body']);
- }
- if (isset($options['status'])) {
- $this->statusCode($options['status']);
- }
- if (isset($options['type'])) {
- $this->type($options['type']);
- }
- if (!isset($options['charset'])) {
- $options['charset'] = Configure::read('App.encoding');
- }
- $this->charset($options['charset']);
- }
- /**
- * Sends the complete response to the client including headers and message body.
- * Will echo out the content in the response body.
- *
- * @return void
- */
- public function send() {
- if (isset($this->_headers['Location']) && $this->_status === 200) {
- $this->statusCode(302);
- }
- $codeMessage = $this->_statusCodes[$this->_status];
- $this->_setCookies();
- $this->_sendHeader("{$this->_protocol} {$this->_status} {$codeMessage}");
- $this->_setContent();
- $this->_setContentLength();
- $this->_setContentType();
- foreach ($this->_headers as $header => $value) {
- $this->_sendHeader($header, $value);
- }
- if ($this->_file) {
- $this->_sendFile($this->_file);
- $this->_file = null;
- } else {
- $this->_sendContent($this->_body);
- }
- }
- /**
- * Sets the cookies that have been added via static method CakeResponse::addCookie()
- * before any other output is sent to the client.
- * Will set the cookies in the order they have been set.
- *
- * @return void
- */
- protected function _setCookies() {
- foreach ($this->_cookies as $name => $c) {
- setcookie(
- $name, $c['value'], $c['expire'], $c['path'],
- $c['domain'], $c['secure'], $c['httpOnly']
- );
- }
- }
- /**
- * 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, array(304, 204))) {
- return;
- }
- $whitelist = array(
- '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->header('Content-Type', "{$this->_contentType}; charset={$this->_charset}");
- } else {
- $this->header('Content-Type', "{$this->_contentType}");
- }
- }
- /**
- * Sets the response body to an empty text if the status code is 204 or 304
- *
- * @return void
- */
- protected function _setContent() {
- if (in_array($this->_status, array(304, 204))) {
- $this->body('');
- }
- }
- /**
- * Calculates the correct Content-Length and sets it as a header in the response
- * Will not set the value if already set or if the output is compressed.
- *
- * @return void
- */
- protected function _setContentLength() {
- $shouldSetLength = !isset($this->_headers['Content-Length']) && !in_array($this->_status, range(301, 307));
- if (isset($this->_headers['Content-Length']) && $this->_headers['Content-Length'] === false) {
- unset($this->_headers['Content-Length']);
- return;
- }
- if ($shouldSetLength && !$this->outputCompressed()) {
- $offset = ob_get_level() ? ob_get_length() : 0;
- if (ini_get('mbstring.func_overload') & 2 && function_exists('mb_strlen')) {
- $this->length($offset + mb_strlen($this->_body, '8bit'));
- } else {
- $this->length($this->_headers['Content-Length'] = $offset + strlen($this->_body));
- }
- }
- }
- /**
- * Sends a header to the client.
- *
- * @param string $name the header name
- * @param string $value the header value
- * @return void
- */
- protected function _sendHeader($name, $value = null) {
- if (!headers_sent()) {
- if (is_null($value)) {
- header($name);
- } else {
- header("{$name}: {$value}");
- }
- }
- }
- /**
- * Sends a content string to the client.
- *
- * @param string $content string to send as response body
- * @return void
- */
- protected function _sendContent($content) {
- echo $content;
- }
- /**
- * Buffers a header string to be sent
- * Returns the complete list of buffered headers
- *
- * ### Single header
- * e.g `header('Location', 'http://example.com');`
- *
- * ### Multiple headers
- * e.g `header(array('Location' => 'http://example.com', 'X-Extra' => 'My header'));`
- *
- * ### String header
- * e.g `header('WWW-Authenticate: Negotiate');`
- *
- * ### Array of string headers
- * e.g `header(array('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
- * e.g `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 $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 $value. The header value.
- * @return array list of headers to be sent
- */
- public function header($header = null, $value = null) {
- if (is_null($header)) {
- return $this->_headers;
- }
- if (is_array($header)) {
- foreach ($header as $h => $v) {
- if (is_numeric($h)) {
- $this->header($v);
- continue;
- }
- $this->_headers[$h] = trim($v);
- }
- return $this->_headers;
- }
- if (!is_null($value)) {
- $this->_headers[$header] = $value;
- return $this->_headers;
- }
- list($header, $value) = explode(':', $header, 2);
- $this->_headers[$header] = trim($value);
- return $this->_headers;
- }
- /**
- * Buffers the response message to be sent
- * if $content is null the current buffer is returned
- *
- * @param string $content the string message to be sent
- * @return string current message buffer if $content param is passed as null
- */
- public function body($content = null) {
- if (is_null($content)) {
- return $this->_body;
- }
- return $this->_body = $content;
- }
- /**
- * Sets the HTTP status code to be sent
- * if $code is null the current code is returned
- *
- * @param integer $code
- * @return integer current status code
- * @throws CakeException When an unknown status code is reached.
- */
- public function statusCode($code = null) {
- if (is_null($code)) {
- return $this->_status;
- }
- if (!isset($this->_statusCodes[$code])) {
- throw new CakeException(__d('cake_dev', 'Unknown status code'));
- }
- return $this->_status = $code;
- }
- /**
- * Queries & sets valid HTTP response codes & messages.
- *
- * @param integer|array $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 'code' and 'message' keys of each nested array are added to the default
- * HTTP codes. Example:
- *
- * httpCodes(404); // returns array(404 => 'Not Found')
- *
- * httpCodes(array(
- * 701 => 'Unicorn Moved',
- * 800 => 'Unexpected Minotaur'
- * )); // sets these new values, and returns true
- *
- * @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.
- */
- public function httpCodes($code = null) {
- if (empty($code)) {
- return $this->_statusCodes;
- }
- if (is_array($code)) {
- $this->_statusCodes = $code + $this->_statusCodes;
- return true;
- }
- if (!isset($this->_statusCodes[$code])) {
- return null;
- }
- return array($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
- *
- * e.g `type('jpg');`
- *
- * ### Returning the current content type
- *
- * e.g `type();`
- *
- * ### Storing content type definitions
- *
- * e.g `type(array('keynote' => 'application/keynote', 'bat' => 'application/bat'));`
- *
- * ### Replacing a content type definition
- *
- * e.g `type(array('jpg' => 'text/plain'));`
- *
- * @param string $contentType
- * @return mixed current content type or false if supplied an invalid content type
- */
- public function type($contentType = null) {
- if (is_null($contentType)) {
- return $this->_contentType;
- }
- if (is_array($contentType)) {
- foreach ($contentType as $type => $definition) {
- $this->_mimeTypes[$type] = $definition;
- }
- return $this->_contentType;
- }
- if (isset($this->_mimeTypes[$contentType])) {
- $contentType = $this->_mimeTypes[$contentType];
- $contentType = is_array($contentType) ? current($contentType) : $contentType;
- }
- if (strpos($contentType, '/') === false) {
- return false;
- }
- return $this->_contentType = $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 mixed Aliases for the types provided.
- */
- public function mapType($ctype) {
- if (is_array($ctype)) {
- return array_map(array($this, 'mapType'), $ctype);
- }
- foreach ($this->_mimeTypes as $alias => $types) {
- if (is_array($types) && in_array($ctype, $types)) {
- return $alias;
- } elseif (is_string($types) && $types == $ctype) {
- return $alias;
- }
- }
- return null;
- }
- /**
- * Sets the response charset
- * if $charset is null the current charset is returned
- *
- * @param string $charset
- * @return string current charset
- */
- public function charset($charset = null) {
- if (is_null($charset)) {
- return $this->_charset;
- }
- return $this->_charset = $charset;
- }
- /**
- * Sets the correct headers to instruct the client to not cache the response
- *
- * @return void
- */
- public function disableCache() {
- $this->header(array(
- 'Expires' => 'Mon, 26 Jul 1997 05:00:00 GMT',
- 'Last-Modified' => gmdate("D, d M Y H:i:s") . " GMT",
- '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
- */
- public function cache($since, $time = '+1 day') {
- if (!is_int($time)) {
- $time = strtotime($time);
- }
- $this->header(array(
- 'Date' => gmdate("D, j M Y G:i:s ", time()) . 'GMT'
- ));
- $this->modified($since);
- $this->expires($time);
- $this->sharable(true);
- $this->maxAge($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 boolean $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 integer $time time in seconds after which the response should no longer be considered fresh
- * @return boolean
- */
- 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;
- }
- $sharable = $public || ! ($private || $noCache);
- return $sharable;
- }
- if ($public) {
- $this->_cacheDirectives['public'] = true;
- unset($this->_cacheDirectives['private']);
- $this->sharedMaxAge($time);
- } else {
- $this->_cacheDirectives['private'] = true;
- unset($this->_cacheDirectives['public']);
- $this->maxAge($time);
- }
- if (!$time) {
- $this->_setCacheControl();
- }
- return (bool)$public;
- }
- /**
- * 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 integer $seconds if null, the method will return the current s-maxage value
- * @return int
- */
- 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;
- }
- /**
- * 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 integer $seconds if null, the method will return the current max-age value
- * @return int
- */
- 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;
- }
- /**
- * Sets the Cache-Control must-revalidate directive.
- * must-revalidate indicates that the response should not be served
- * stale by a cache under any cirumstance without first revalidating
- * with the origin.
- * If called with no parameters, this function will return wheter must-revalidate is present.
- *
- * @param integer $seconds if null, the method will return the current
- * must-revalidate value
- * @return boolean
- */
- public function mustRevalidate($enable = null) {
- 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);
- }
- /**
- * 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->header('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 $time
- * @return string
- */
- public function expires($time = null) {
- if ($time !== null) {
- $date = $this->_getUTCDate($time);
- $this->_headers['Expires'] = $date->format('D, j M Y H:i:s') . ' GMT';
- }
- if (isset($this->_headers['Expires'])) {
- return $this->_headers['Expires'];
- }
- return null;
- }
- /**
- * Sets the Last-Modified header for the response by taking an 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 $time
- * @return string
- */
- public function modified($time = null) {
- if ($time !== null) {
- $date = $this->_getUTCDate($time);
- $this->_headers['Last-Modified'] = $date->format('D, j M Y H:i:s') . ' GMT';
- }
- if (isset($this->_headers['Last-Modified'])) {
- return $this->_headers['Last-Modified'];
- }
- return null;
- }
- /**
- * Sets the response as Not Modified by removing any body contents
- * setting the status code to "304 Not Modified" and removing all
- * conflicting headers
- *
- * @return void
- */
- public function notModified() {
- $this->statusCode(304);
- $this->body('');
- $remove = array(
- 'Allow',
- 'Content-Encoding',
- 'Content-Language',
- 'Content-Length',
- 'Content-MD5',
- 'Content-Type',
- 'Last-Modified'
- );
- foreach ($remove as $header) {
- unset($this->_headers[$header]);
- }
- }
- /**
- * 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 $cacheVariances a single Vary string or a array
- * containig the list for variances.
- * @return array
- */
- public function vary($cacheVariances = null) {
- if ($cacheVariances !== null) {
- $cacheVariances = (array)$cacheVariances;
- $this->_headers['Vary'] = implode(', ', $cacheVariances);
- }
- if (isset($this->_headers['Vary'])) {
- return explode(', ', $this->_headers['Vary']);
- }
- return null;
- }
- /**
- * Sets the response Etag, Etags are a strong indicative that a response
- * can be cached by a HTTP client. A bad way of generaing 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 instuct clients that the content has
- * changed, but sematicallly, 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 $hash the unique has that identifies this resposnse
- * @param boolean $weak whether the response is semantically the same as
- * other with th same hash or not
- * @return string
- */
- public function etag($tag = null, $weak = false) {
- if ($tag !== null) {
- $this->_headers['Etag'] = sprintf('%s"%s"', ($weak) ? 'W/' : null, $tag);
- }
- if (isset($this->_headers['Etag'])) {
- return $this->_headers['Etag'];
- }
- return null;
- }
- /**
- * Returns a DateTime object initialized at the $time param and using UTC
- * as timezone
- *
- * @param string|integer|DateTime $time
- * @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 boolean 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 boolean
- */
- 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
- */
- public function download($filename) {
- $this->header('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 protocol to be used for sending response
- * @return string protocol currently set
- */
- public function protocol($protocol = null) {
- 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 integer $bytes Number of bytes
- * @return integer|null
- */
- public function length($bytes = null) {
- if ($bytes !== null) {
- $this->_headers['Content-Length'] = $bytes;
- }
- if (isset($this->_headers['Content-Length'])) {
- return $this->_headers['Content-Length'];
- }
- return null;
- }
- /**
- * Checks whether a response has not been modified according to the 'If-None-Match'
- * (Etags) and 'If-Modified-Since' (last modification date) request
- * headers 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.
- *
- * @param CakeRequest $request Request object
- * @return boolean whether the response was marked as not modified or not.
- */
- public function checkNotModified(CakeRequest $request) {
- $etags = preg_split('/\s*,\s*/', $request->header('If-None-Match'), null, PREG_SPLIT_NO_EMPTY);
- $modifiedSince = $request->header('If-Modified-Since');
- if ($responseTag = $this->etag()) {
- $etagMatches = in_array('*', $etags) || in_array($responseTag, $etags);
- }
- if ($modifiedSince) {
- $timeMatches = strtotime($this->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.
- *
- * @return string
- */
- public function __toString() {
- return (string)$this->_body;
- }
- /**
- * 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.
- *
- * @param array $options Either null to get all cookies, string for a specific cookie
- * or array to set cookie.
- *
- * ### 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)`
- *
- * @return mixed
- */
- public function cookie($options = null) {
- if ($options === null) {
- return $this->_cookies;
- }
- if (is_string($options)) {
- if (!isset($this->_cookies[$options])) {
- return null;
- }
- return $this->_cookies[$options];
- }
- $defaults = array(
- 'name' => 'CakeCookie[default]',
- 'value' => '',
- 'expire' => 0,
- 'path' => '/',
- 'domain' => '',
- 'secure' => false,
- 'httpOnly' => false
- );
- $options += $defaults;
- $this->_cookies[$options['name']] = $options;
- }
- /**
- * Setup for display or download the given file
- *
- * @param string $path Path to file
- * @param array $options Options
- * ### Options keys
- * - name: Alternate download name
- * - download: If `true` sets download header and forces file to be downloaded rather than displayed in browser
- * @return void
- * @throws NotFoundException
- */
- public function file($path, $options = array()) {
- $options += array(
- 'name' => null,
- 'download' => null
- );
- if (!is_file($path)) {
- $path = APP . $path;
- }
- $file = new File($path);
- if (!$file->exists() || !$file->readable()) {
- if (Configure::read('debug')) {
- throw new NotFoundException(__d('cake_dev', 'The requested file %s was not found or not readable', $path));
- }
- throw new NotFoundException(__d('cake', 'The requested file was not found'));
- }
- $extension = strtolower($file->ext());
- $download = $options['download'];
- if ((!$extension || $this->type($extension) === false) && is_null($download)) {
- $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/octetstream';
- } elseif (preg_match('/MSIE ([0-9].[0-9]{1,2})/', $agent)) {
- $contentType = 'application/force-download';
- }
- if (!empty($contentType)) {
- $this->type($contentType);
- }
- if (is_null($options['name'])) {
- $name = $file->name;
- } else {
- $name = $options['name'];
- }
- $this->download($name);
- $this->header('Accept-Ranges', 'bytes');
- $httpRange = env('HTTP_RANGE');
- if (isset($httpRange)) {
- list(, $range) = explode('=', $httpRange);
- $size = $fileSize - 1;
- $length = $fileSize - $range;
- $this->header(array(
- 'Content-Length' => $length,
- 'Content-Range' => 'bytes ' . $range . $size . '/' . $fileSize
- ));
- $this->statusCode(206);
- $file->open('rb', true);
- $file->offset($range);
- } else {
- $this->header('Content-Length', $fileSize);
- }
- } else {
- $this->header('Content-Length', $fileSize);
- }
- $this->_clearBuffer();
- $this->_file = $file;
- }
- /**
- * Reads out a file, and echos the content to the client.
- *
- * @param File $file File object
- * @return boolean True is whole file is echoed successfully or false if client connection is lost in between
- */
- protected function _sendFile($file) {
- $compress = $this->outputCompressed();
- $file->open('rb');
- while (!feof($file->handle)) {
- if (!$this->_isActive()) {
- $file->close();
- return false;
- }
- set_time_limit(0);
- echo fread($file->handle, 8192);
- if (!$compress) {
- $this->_flushBuffer();
- }
- }
- $file->close();
- return true;
- }
- /**
- * Returns true if connection is still active
- *
- * @return boolean
- */
- protected function _isActive() {
- return connection_status() === CONNECTION_NORMAL && !connection_aborted();
- }
- /**
- * Clears the contents of the topmost output buffer and discards them
- *
- * @return boolean
- */
- protected function _clearBuffer() {
- //@codingStandardsIgnoreStart
- return @ob_end_clean();
- //@codingStandardsIgnoreEnd
- }
- /**
- * Flushes the contents of the output buffer
- *
- * @return void
- */
- protected function _flushBuffer() {
- //@codingStandardsIgnoreStart
- @flush();
- @ob_flush();
- //@codingStandardsIgnoreEnd
- }
- }
|