Response.php 20.1 KB
Newer Older
Qiang Xue committed
1 2 3
<?php
/**
 * @link http://www.yiiframework.com/
Qiang Xue committed
4
 * @copyright Copyright (c) 2008 Yii Software LLC
Qiang Xue committed
5 6 7 8 9
 * @license http://www.yiiframework.com/license/
 */

namespace yii\web;

Qiang Xue committed
10
use Yii;
11
use yii\base\InvalidConfigException;
Qiang Xue committed
12
use yii\base\InvalidParamException;
Qiang Xue committed
13
use yii\helpers\FileHelper;
Qiang Xue committed
14
use yii\helpers\Html;
Qiang Xue committed
15
use yii\helpers\Json;
Qiang Xue committed
16
use yii\helpers\SecurityHelper;
17
use yii\helpers\StringHelper;
Qiang Xue committed
18 19 20

/**
 * @author Qiang Xue <qiang.xue@gmail.com>
21
 * @author Carsten Brandt <mail@cebe.cc>
Qiang Xue committed
22 23 24 25
 * @since 2.0
 */
class Response extends \yii\base\Response
{
26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
	const FORMAT_RAW = 'raw';
	const FORMAT_HTML = 'html';
	const FORMAT_JSON = 'json';
	const FORMAT_JSONP = 'jsonp';
	const FORMAT_XML = 'xml';

	/**
	 * @var string the response format.
	 */
	public $format = self::FORMAT_HTML;
	/**
	 * @var string the charset of the text response. If not set, it will use
	 * the value of [[Application::charset]].
	 */
	public $charset;
Qiang Xue committed
41 42 43 44 45 46 47
	/**
	 * @var integer the HTTP status code that should be used when redirecting in AJAX mode.
	 * This is used by [[redirect()]]. A 2xx code should normally be used for this purpose
	 * so that the AJAX handler will treat the response as a success.
	 * @see redirect
	 */
	public $ajaxRedirectCode = 278;
Qiang Xue committed
48 49 50 51 52
	/**
	 * @var string
	 */
	public $statusText;
	/**
53 54
	 * @var string the version of the HTTP protocol to use. If not set, it will be determined via `$_SERVER['SERVER_PROTOCOL']`,
	 * or '1.1' if that is not available.
Qiang Xue committed
55
	 */
56
	public $version;
Qiang Xue committed
57 58 59
	/**
	 * @var array list of HTTP status codes and the corresponding texts
	 */
Qiang Xue committed
60
	public static $httpStatuses = array(
Qiang Xue committed
61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103
		100 => 'Continue',
		101 => 'Switching Protocols',
		102 => 'Processing',
		118 => 'Connection timed out',
		200 => 'OK',
		201 => 'Created',
		202 => 'Accepted',
		203 => 'Non-Authoritative',
		204 => 'No Content',
		205 => 'Reset Content',
		206 => 'Partial Content',
		207 => 'Multi-Status',
		208 => 'Already Reported',
		210 => 'Content Different',
		226 => 'IM Used',
		300 => 'Multiple Choices',
		301 => 'Moved Permanently',
		302 => 'Found',
		303 => 'See Other',
		304 => 'Not Modified',
		305 => 'Use Proxy',
		306 => 'Reserved',
		307 => 'Temporary Redirect',
		308 => 'Permanent Redirect',
		310 => 'Too many 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 Long',
		415 => 'Unsupported Media Type',
		416 => 'Requested range unsatisfiable',
		417 => 'Expectation failed',
104
		418 => 'I\'m a teapot',
Qiang Xue committed
105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127
		422 => 'Unprocessable entity',
		423 => 'Locked',
		424 => 'Method failure',
		425 => 'Unordered Collection',
		426 => 'Upgrade Required',
		428 => 'Precondition Required',
		429 => 'Too Many Requests',
		431 => 'Request Header Fields Too Large',
		449 => 'Retry With',
		450 => 'Blocked by Windows Parental Controls',
		500 => 'Internal Server Error',
		501 => 'Not Implemented',
		502 => 'Bad Gateway ou Proxy Error',
		503 => 'Service Unavailable',
		504 => 'Gateway Time-out',
		505 => 'HTTP Version not supported',
		507 => 'Insufficient storage',
		508 => 'Loop Detected',
		509 => 'Bandwidth Limit Exceeded',
		510 => 'Not Extended',
		511 => 'Network Authentication Required',
	);

128 129 130
	/**
	 * @var integer the HTTP status code to send with the response.
	 */
Qiang Xue committed
131
	private $_statusCode;
Qiang Xue committed
132 133 134
	/**
	 * @var HeaderCollection
	 */
Qiang Xue committed
135 136
	private $_headers;

Qiang Xue committed
137 138 139

	public function init()
	{
140 141 142 143 144 145
		if ($this->version === null) {
			if (isset($_SERVER['SERVER_PROTOCOL']) && $_SERVER['SERVER_PROTOCOL'] === '1.0') {
				$this->version = '1.0';
			} else {
				$this->version = '1.1';
			}
Qiang Xue committed
146
		}
147 148 149
		if ($this->charset === null) {
			$this->charset = Yii::$app->charset;
		}
Qiang Xue committed
150 151
	}

152 153 154
	/**
	 * @return integer the HTTP status code to send with the response.
	 */
Qiang Xue committed
155 156 157 158 159
	public function getStatusCode()
	{
		return $this->_statusCode;
	}

Qiang Xue committed
160
	public function setStatusCode($value, $text = null)
Qiang Xue committed
161 162
	{
		$this->_statusCode = (int)$value;
163
		if ($this->getIsInvalid()) {
Qiang Xue committed
164 165
			throw new InvalidParamException("The HTTP status code is invalid: $value");
		}
Qiang Xue committed
166
		if ($text === null) {
Qiang Xue committed
167
			$this->statusText = isset(self::$httpStatuses[$this->_statusCode]) ? self::$httpStatuses[$this->_statusCode] : '';
Qiang Xue committed
168 169 170
		} else {
			$this->statusText = $text;
		}
Qiang Xue committed
171 172
	}

Qiang Xue committed
173 174 175 176 177 178 179 180 181 182 183 184 185
	/**
	 * Returns the header collection.
	 * The header collection contains the currently registered HTTP headers.
	 * @return HeaderCollection the header collection
	 */
	public function getHeaders()
	{
		if ($this->_headers === null) {
			$this->_headers = new HeaderCollection;
		}
		return $this->_headers;
	}

Qiang Xue committed
186 187 188 189 190
	/**
	 * Sends the response to the client.
	 */
	public function send()
	{
191
		parent::send();
Qiang Xue committed
192 193
		$this->sendHeaders();
		$this->sendContent();
194
		$this->clear();
Qiang Xue committed
195 196
	}

197
	public function clear()
Qiang Xue committed
198 199 200
	{
		$this->_headers = null;
		$this->_statusCode = null;
201
		$this->_content = null;
Qiang Xue committed
202 203 204
		$this->statusText = null;
	}

Qiang Xue committed
205 206 207 208 209
	/**
	 * Sends the response headers to the client
	 */
	protected function sendHeaders()
	{
Qiang Xue committed
210 211 212
		if (headers_sent()) {
			return;
		}
Qiang Xue committed
213 214 215 216
		$statusCode = $this->getStatusCode();
		if ($statusCode !== null) {
			header("HTTP/{$this->version} $statusCode {$this->statusText}");
		}
Qiang Xue committed
217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243
		if ($this->_headers) {
			$headers = $this->getHeaders();
			foreach ($headers as $name => $values) {
				foreach ($values as $value) {
					header("$name: $value", false);
				}
			}
		}
		$this->sendCookies();
	}

	/**
	 * Sends the cookies to the client.
	 */
	protected function sendCookies()
	{
		if ($this->_cookies === null) {
			return;
		}
		$request = Yii::$app->getRequest();
		if ($request->enableCookieValidation) {
			$validationKey = $request->getCookieValidationKey();
		}
		foreach ($this->getCookies() as $cookie) {
			$value = $cookie->value;
			if ($cookie->expire != 1  && isset($validationKey)) {
				$value = SecurityHelper::hashData(serialize($value), $validationKey);
Qiang Xue committed
244
			}
Qiang Xue committed
245
			setcookie($cookie->name, $value, $cookie->expire, $cookie->path, $cookie->domain, $cookie->secure, $cookie->httpOnly);
Qiang Xue committed
246
		}
Qiang Xue committed
247
		$this->getCookies()->removeAll();
Qiang Xue committed
248 249 250 251 252 253 254
	}

	/**
	 * Sends the response content to the client
	 */
	protected function sendContent()
	{
255
		echo $this->getContent();
Qiang Xue committed
256 257
	}

Qiang Xue committed
258
	/**
259 260 261
	 * Sends a file to the browser.
	 * @param string $filePath the path of the file to be sent.
	 * @param string $attachmentName the file name shown to the user. If null, it will be determined from `$filePath`.
Qiang Xue committed
262
	 * @param string $mimeType the MIME type of the content. If null, it will be guessed based on `$filePath`
Qiang Xue committed
263
	 */
Qiang Xue committed
264
	public function sendFile($filePath, $attachmentName = null, $mimeType = null)
Qiang Xue committed
265
	{
266
		if ($mimeType === null && ($mimeType = FileHelper::getMimeTypeByExtension($filePath)) === null) {
267
			$mimeType = 'application/octet-stream';
Qiang Xue committed
268
		}
269 270
		if ($attachmentName === null) {
			$attachmentName = basename($filePath);
271
		}
272
		$handle = fopen($filePath, 'rb');
Qiang Xue committed
273
		$this->sendStreamAsFile($handle, $attachmentName, $mimeType);
274
	}
275

276 277 278 279
	/**
	 * Sends the specified content as a file to the browser.
	 * @param string $content the content to be sent. The existing [[content]] will be discarded.
	 * @param string $attachmentName the file name shown to the user.
Qiang Xue committed
280
	 * @param string $mimeType the MIME type of the content.
281
	 * @throws HttpException if the requested range is not satisfiable
282
	 */
283
	public function sendContentAsFile($content, $attachmentName, $mimeType = 'application/octet-stream')
284
	{
285 286 287 288 289 290 291 292
		$headers = $this->getHeaders();
		$contentLength = StringHelper::strlen($content);
		$range = $this->getHttpRange($contentLength);
		if ($range === false) {
			$headers->set('Content-Range', "bytes */$contentLength");
			throw new HttpException(416, Yii::t('yii', 'Requested range not satisfiable'));
		}

293 294 295 296 297 298 299 300
		$headers->setDefault('Pragma', 'public')
			->setDefault('Accept-Ranges', 'bytes')
			->setDefault('Expires', '0')
			->setDefault('Content-Type', $mimeType)
			->setDefault('Cache-Control', 'must-revalidate, post-check=0, pre-check=0')
			->setDefault('Content-Transfer-Encoding', 'binary')
			->setDefault('Content-Length', StringHelper::strlen($content))
			->setDefault('Content-Disposition', "attachment; filename=\"$attachmentName\"");
301

302 303 304 305
		list($begin, $end) = $range;
		if ($begin !=0 || $end != $contentLength - 1) {
			$this->setStatusCode(206);
			$headers->set('Content-Range', "bytes $begin-$end/$contentLength");
306
			$this->setContent(StringHelper::substr($content, $begin, $end - $begin + 1), self::FORMAT_RAW);
307 308
		} else {
			$this->setStatusCode(200);
309
			$this->setContent($content, self::FORMAT_RAW);
310 311
		}

312
		$this->send();
Qiang Xue committed
313 314
	}

315 316 317 318
	/**
	 * Sends the specified stream as a file to the browser.
	 * @param resource $handle the handle of the stream to be sent.
	 * @param string $attachmentName the file name shown to the user.
Qiang Xue committed
319
	 * @param string $mimeType the MIME type of the stream content.
320 321
	 * @throws HttpException if the requested range cannot be satisfied.
	 */
322
	public function sendStreamAsFile($handle, $attachmentName, $mimeType = 'application/octet-stream')
Qiang Xue committed
323
	{
324
		$headers = $this->getHeaders();
Qiang Xue committed
325 326 327
		fseek($handle, 0, SEEK_END);
		$fileSize = ftell($handle);

328 329 330 331 332
		$range = $this->getHttpRange($fileSize);
		if ($range === false) {
			$headers->set('Content-Range', "bytes */$fileSize");
			throw new HttpException(416, Yii::t('yii', 'Requested range not satisfiable'));
		}
Qiang Xue committed
333

334 335
		list($begin, $end) = $range;
		if ($begin !=0 || $end != $fileSize - 1) {
Qiang Xue committed
336
			$this->setStatusCode(206);
337
			$headers->set('Content-Range', "bytes $begin-$end/$fileSize");
Qiang Xue committed
338 339 340 341
		} else {
			$this->setStatusCode(200);
		}

342
		$length = $end - $begin + 1;
Qiang Xue committed
343

344 345 346 347 348 349 350 351
		$headers->setDefault('Pragma', 'public')
			->setDefault('Accept-Ranges', 'bytes')
			->setDefault('Expires', '0')
			->setDefault('Content-Type', $mimeType)
			->setDefault('Cache-Control', 'must-revalidate, post-check=0, pre-check=0')
			->setDefault('Content-Transfer-Encoding', 'binary')
			->setDefault('Content-Length', $length)
			->setDefault('Content-Disposition', "attachment; filename=\"$attachmentName\"");
Qiang Xue committed
352

353
		$this->send();
Qiang Xue committed
354

355
		fseek($handle, $begin);
Qiang Xue committed
356 357
		set_time_limit(0); // Reset time limit for big files
		$chunkSize = 8 * 1024 * 1024; // 8MB per chunk
358 359 360
		while (!feof($handle) && ($pos = ftell($handle)) <= $end) {
			if ($pos + $chunkSize > $end) {
				$chunkSize = $end - $pos + 1;
Qiang Xue committed
361 362 363 364 365 366 367
			}
			echo fread($handle, $chunkSize);
			flush(); // Free up memory. Otherwise large files will trigger PHP's memory limit.
		}
		fclose($handle);
	}

368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400
	/**
	 * Determines the HTTP range given in the request.
	 * @param integer $fileSize the size of the file that will be used to validate the requested HTTP range.
	 * @return array|boolean the range (begin, end), or false if the range request is invalid.
	 */
	protected function getHttpRange($fileSize)
	{
		if (!isset($_SERVER['HTTP_RANGE']) || $_SERVER['HTTP_RANGE'] === '-') {
			return array(0, $fileSize - 1);
		}
		if (!preg_match('/^bytes=(\d*)-(\d*)$/', $_SERVER['HTTP_RANGE'], $matches)) {
			return false;
		}
		if ($matches[1] === '') {
			$start = $fileSize - $matches[2];
			$end = $fileSize - 1;
		} elseif ($matches[2] !== '') {
			$start = $matches[1];
			$end = $matches[2];
			if ($end >= $fileSize) {
				$end = $fileSize - 1;
			}
		} else {
			$start = $matches[1];
			$end = $fileSize - 1;
		}
		if ($start < 0 || $start > $end) {
			return false;
		} else {
			return array($start, $end);
		}
	}

Qiang Xue committed
401 402 403 404 405 406 407 408 409 410 411 412 413 414
	/**
	 * Sends existing file to a browser as a download using x-sendfile.
	 *
	 * X-Sendfile is a feature allowing a web application to redirect the request for a file to the webserver
	 * that in turn processes the request, this way eliminating the need to perform tasks like reading the file
	 * and sending it to the user. When dealing with a lot of files (or very big files) this can lead to a great
	 * increase in performance as the web application is allowed to terminate earlier while the webserver is
	 * handling the request.
	 *
	 * The request is sent to the server through a special non-standard HTTP-header.
	 * When the web server encounters the presence of such header it will discard all output and send the file
	 * specified by that header using web server internals including all optimizations like caching-headers.
	 *
	 * As this header directive is non-standard different directives exists for different web servers applications:
Qiang Xue committed
415 416 417 418 419 420 421
	 * 
	 * - Apache: [X-Sendfile](http://tn123.org/mod_xsendfile) 
	 * - Lighttpd v1.4: [X-LIGHTTPD-send-file](http://redmine.lighttpd.net/projects/lighttpd/wiki/X-LIGHTTPD-send-file)
	 * - Lighttpd v1.5: [X-Sendfile](http://redmine.lighttpd.net/projects/lighttpd/wiki/X-LIGHTTPD-send-file)
	 * - Nginx: [X-Accel-Redirect](http://wiki.nginx.org/XSendfile)
	 * - Cherokee: [X-Sendfile and X-Accel-Redirect](http://www.cherokee-project.com/doc/other_goodies.html#x-sendfile)
	 *
Qiang Xue committed
422 423 424
	 * So for this method to work the X-SENDFILE option/module should be enabled by the web server and
	 * a proper xHeader should be sent.
	 *
Qiang Xue committed
425 426 427 428
	 * **Note**
	 * 
	 * This option allows to download files that are not under web folders, and even files that are otherwise protected 
	 * (deny from all) like `.htaccess`.
Qiang Xue committed
429
	 *
Qiang Xue committed
430 431
	 * **Side effects**
	 * 
Qiang Xue committed
432 433 434
	 * If this option is disabled by the web server, when this method is called a download configuration dialog
	 * will open but the downloaded file will have 0 bytes.
	 *
Qiang Xue committed
435 436
	 * **Known issues**
	 * 
Qiang Xue committed
437
	 * There is a Bug with Internet Explorer 6, 7 and 8 when X-SENDFILE is used over an SSL connection, it will show
Qiang Xue committed
438 439 440 441 442 443
	 * an error message like this: "Internet Explorer was not able to open this Internet site. The requested site 
	 * is either unavailable or cannot be found.". You can work around this problem by removing the `Pragma`-header.
	 *
	 * **Example**
	 * 
	 * ~~~
444 445
	 * Yii::app()->request->xSendFile('/home/user/Pictures/picture1.jpg');
	 * ~~~
Qiang Xue committed
446
	 *
Qiang Xue committed
447
	 * @param string $filePath file name with full path
448 449 450
	 * @param string $mimeType the MIME type of the file. If null, it will be determined based on `$filePath`.
	 * @param string $attachmentName file name shown to the user. If null, it will be determined from `$filePath`.
	 * @param string $xHeader the name of the x-sendfile header.
Qiang Xue committed
451
	 */
Qiang Xue committed
452
	public function xSendFile($filePath, $attachmentName = null, $mimeType = null, $xHeader = 'X-Sendfile')
Qiang Xue committed
453
	{
454 455
		if ($mimeType === null && ($mimeType = FileHelper::getMimeTypeByExtension($filePath)) === null) {
			$mimeType = 'application/octet-stream';
Qiang Xue committed
456
		}
457 458
		if ($attachmentName === null) {
			$attachmentName = basename($filePath);
Qiang Xue committed
459
		}
Qiang Xue committed
460

461
		$this->getHeaders()
462 463 464
			->setDefault($xHeader, $filePath)
			->setDefault('Content-Type', $mimeType)
			->setDefault('Content-Disposition', "attachment; filename=\"$attachmentName\"");
Qiang Xue committed
465

Qiang Xue committed
466
		$this->send();
Qiang Xue committed
467
	}
Qiang Xue committed
468 469 470

	/**
	 * Redirects the browser to the specified URL.
Qiang Xue committed
471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488
	 * This method will send out a "Location" header to achieve the redirection.
	 * In AJAX mode, this normally will not work as expected unless there are some
	 * client-side JavaScript code handling the redirection. To help achieve this goal,
	 * this method will use [[ajaxRedirectCode]] as the HTTP status code when performing
	 * redirection in AJAX mode. The following JavaScript code may be used on the client
	 * side to handle the redirection response:
	 *
	 * ~~~
	 * $(document).ajaxSuccess(function(event, xhr, settings) {
	 *     if (xhr.status == 278) {
	 *         window.location = xhr.getResponseHeader('Location');
	 *     }
	 * });
	 * ~~~
	 *
	 * @param array|string $url the URL to be redirected to. [[\yii\helpers\Html::url()]]
	 * will be used to normalize the URL. If the resulting URL is still a relative URL
	 * (one without host info), the current request host info will be used.
489 490
	 * @param integer $statusCode the HTTP status code. If null, it will use 302
	 * for normal requests, and [[ajaxRedirectCode]] for AJAX requests.
Qiang Xue committed
491
	 * See [[http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html]]
492
	 * for details about HTTP status code
493
	 * @return Response the response object itself
Qiang Xue committed
494
	 */
495
	public function redirect($url, $statusCode = null)
Qiang Xue committed
496
	{
Qiang Xue committed
497
		$url = Html::url($url);
Qiang Xue committed
498 499 500
		if (strpos($url, '/') === 0 && strpos($url, '//') !== 0) {
			$url = Yii::$app->getRequest()->getHostInfo() . $url;
		}
501 502
		if ($statusCode === null) {
			$statusCode = Yii::$app->getRequest()->getIsAjax() ? $this->ajaxRedirectCode : 302;
Qiang Xue committed
503
		}
Qiang Xue committed
504 505
		$this->getHeaders()->set('Location', $url);
		$this->setStatusCode($statusCode);
506
		return $this;
Qiang Xue committed
507
	}
508

509 510 511 512 513 514
	/**
	 * Refreshes the current page.
	 * The effect of this method call is the same as the user pressing the refresh button of his browser
	 * (without re-posting data).
	 * @param string $anchor the anchor that should be appended to the redirection URL.
	 * Defaults to empty. Make sure the anchor starts with '#' if you want to specify it.
515
	 * @return Response the response object itself
516
	 */
517
	public function refresh($anchor = '')
518
	{
519
		return $this->redirect(Yii::$app->getRequest()->getUrl() . $anchor);
520 521
	}

Qiang Xue committed
522 523
	private $_cookies;

524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544
	/**
	 * Returns the cookie collection.
	 * Through the returned cookie collection, you add or remove cookies as follows,
	 *
	 * ~~~
	 * // add a cookie
	 * $response->cookies->add(new Cookie(array(
	 *     'name' => $name,
	 *     'value' => $value,
	 * ));
	 *
	 * // remove a cookie
	 * $response->cookies->remove('name');
	 * // alternatively
	 * unset($response->cookies['name']);
	 * ~~~
	 *
	 * @return CookieCollection the cookie collection.
	 */
	public function getCookies()
	{
Qiang Xue committed
545 546 547 548
		if ($this->_cookies === null) {
			$this->_cookies = new CookieCollection;
		}
		return $this->_cookies;
549
	}
Qiang Xue committed
550 551 552 553

	/**
	 * @return boolean whether this response has a valid [[statusCode]].
	 */
554
	public function getIsInvalid()
Qiang Xue committed
555 556 557 558 559 560 561
	{
		return $this->getStatusCode() < 100 || $this->getStatusCode() >= 600;
	}

	/**
	 * @return boolean whether this response is informational
	 */
562
	public function getIsInformational()
Qiang Xue committed
563 564 565 566 567
	{
		return $this->getStatusCode() >= 100 && $this->getStatusCode() < 200;
	}

	/**
568
	 * @return boolean whether this response is successful
Qiang Xue committed
569
	 */
570
	public function getIsSuccessful()
Qiang Xue committed
571 572 573 574 575 576 577
	{
		return $this->getStatusCode() >= 200 && $this->getStatusCode() < 300;
	}

	/**
	 * @return boolean whether this response is a redirection
	 */
578
	public function getIsRedirection()
Qiang Xue committed
579 580 581 582 583 584 585
	{
		return $this->getStatusCode() >= 300 && $this->getStatusCode() < 400;
	}

	/**
	 * @return boolean whether this response indicates a client error
	 */
586
	public function getIsClientError()
Qiang Xue committed
587 588 589 590 591 592 593
	{
		return $this->getStatusCode() >= 400 && $this->getStatusCode() < 500;
	}

	/**
	 * @return boolean whether this response indicates a server error
	 */
594
	public function getIsServerError()
Qiang Xue committed
595 596 597 598 599 600 601
	{
		return $this->getStatusCode() >= 500 && $this->getStatusCode() < 600;
	}

	/**
	 * @return boolean whether this response is OK
	 */
602
	public function getIsOk()
Qiang Xue committed
603
	{
604
		return $this->getStatusCode() == 200;
Qiang Xue committed
605 606 607 608 609
	}

	/**
	 * @return boolean whether this response indicates the current request is forbidden
	 */
610
	public function getIsForbidden()
Qiang Xue committed
611
	{
612
		return $this->getStatusCode() == 403;
Qiang Xue committed
613 614 615 616 617
	}

	/**
	 * @return boolean whether this response indicates the currently requested resource is not found
	 */
618
	public function getIsNotFound()
Qiang Xue committed
619
	{
620
		return $this->getStatusCode() == 404;
Qiang Xue committed
621 622 623 624 625
	}

	/**
	 * @return boolean whether this response is empty
	 */
626
	public function getIsEmpty()
Qiang Xue committed
627 628 629
	{
		return in_array($this->getStatusCode(), array(201, 204, 304));
	}
630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669

	private $_content;

	public function getContent()
	{
		return $this->_content;
	}

	public function setContent($value, $format = null)
	{
		if ($format !== null) {
			$this->format = $format;
		}
		$this->_content = $this->formatContent($value, $format);
	}

	protected function formatContent($data, $format)
	{
		switch ($this->format) {
			case self::FORMAT_RAW:
				return $data;
			case self::FORMAT_HTML:
				$this->getHeaders()->setDefault('Content-Type', 'text/html; charset=' . $this->charset);
				return $data;
			case self::FORMAT_JSON:
				$this->getHeaders()->set('Content-Type', 'application/json');
				return Json::encode($data);
			case self::FORMAT_JSONP:
				$this->getHeaders()->set('Content-Type', 'text/javascript');
				if (is_array($data) && isset($data['data'], $data['callback'])) {
					return sprintf('%s(%s);', $data['callback'], Json::encode($data['data']));
				} else {
					throw new InvalidParamException("The 'jsonp' response requires that the data be an array consisting of both 'data' and 'callback' elements.");
				}
			case self::FORMAT_XML:
				// todo
			default:
				throw new InvalidConfigException("Unsupported response format: $format");
		}
	}
Qiang Xue committed
670
}