You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
flight-core/flight/net/Response.php

300 lines
7.2 KiB

14 years ago
<?php
/**
* Flight: An extensible micro-framework.
14 years ago
*
* @copyright Copyright (c) 2011, Mike Cao <mike@mikecao.com>
* @license MIT, http://flightphp.com/license
14 years ago
*/
namespace flight\net;
/**
* The Response class represents an HTTP response. The object
* contains the response headers, HTTP status code, and response
* body.
*/
14 years ago
class Response {
/**
* @var int HTTP status
*/
14 years ago
protected $status = 200;
/**
* @var array HTTP headers
*/
protected $headers = array();
/**
* @var string HTTP response body
*/
14 years ago
protected $body;
/**
* @var bool HTTP response sent
*/
protected $sent = false;
/**
* @var array HTTP status codes
*/
14 years ago
public static $codes = array(
100 => 'Continue',
101 => 'Switching Protocols',
102 => 'Processing',
14 years ago
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',
14 years ago
300 => 'Multiple Choices',
301 => 'Moved Permanently',
302 => 'Found',
303 => 'See Other',
304 => 'Not Modified',
305 => 'Use Proxy',
306 => '(Unused)',
14 years ago
307 => 'Temporary Redirect',
308 => 'Permanent Redirect',
14 years ago
400 => 'Bad Request',
401 => 'Unauthorized',
402 => 'Payment Required',
14 years ago
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 => 'Payload Too Large',
414 => 'URI Too Long',
14 years ago
415 => 'Unsupported Media Type',
416 => 'Range Not Satisfiable',
14 years ago
417 => 'Expectation Failed',
422 => 'Unprocessable Entity',
423 => 'Locked',
424 => 'Failed Dependency',
426 => 'Upgrade Required',
428 => 'Precondition Required',
429 => 'Too Many Requests',
431 => 'Request Header Fields Too Large',
14 years ago
500 => 'Internal Server Error',
501 => 'Not Implemented',
502 => 'Bad Gateway',
503 => 'Service Unavailable',
504 => 'Gateway Timeout',
505 => 'HTTP Version Not Supported',
506 => 'Variant Also Negotiates',
507 => 'Insufficient Storage',
508 => 'Loop Detected',
510 => 'Not Extended',
511 => 'Network Authentication Required'
14 years ago
);
/**
* Sets the HTTP status of the response.
*
* @param int $code HTTP status code.
* @return object|int Self reference
12 years ago
* @throws \Exception If invalid status code
14 years ago
*/
public function status($code = null) {
if ($code === null) {
return $this->status;
}
14 years ago
if (array_key_exists($code, self::$codes)) {
$this->status = $code;
14 years ago
}
else {
throw new \Exception('Invalid status code.');
14 years ago
}
return $this;
}
/**
* Adds a header to the response.
*
12 years ago
* @param string|array $name Header name or array of names and values
14 years ago
* @param string $value Header value
12 years ago
* @return object Self reference
14 years ago
*/
public function header($name, $value = null) {
if (is_array($name)) {
foreach ($name as $k => $v) {
$this->headers[$k] = $v;
}
}
else {
$this->headers[$name] = $value;
}
return $this;
}
/**
* Returns the headers from the response
* @return array
*/
public function headers() {
return $this->headers;
}
14 years ago
/**
* Writes content to the response body.
*
* @param string $str Response content
12 years ago
* @return object Self reference
14 years ago
*/
public function write($str) {
$this->body .= $str;
return $this;
}
/**
* Clears the response.
12 years ago
*
* @return object Self reference
14 years ago
*/
public function clear() {
$this->status = 200;
$this->headers = array();
14 years ago
$this->body = '';
return $this;
}
/**
* Sets caching headers for the response.
*
* @param int|string $expires Expiration time
12 years ago
* @return object Self reference
14 years ago
*/
public function cache($expires) {
if ($expires === false) {
$this->headers['Expires'] = 'Mon, 26 Jul 1997 05:00:00 GMT';
$this->headers['Cache-Control'] = array(
'no-store, no-cache, must-revalidate',
'post-check=0, pre-check=0',
'max-age=0'
);
$this->headers['Pragma'] = 'no-cache';
}
else {
$expires = is_int($expires) ? $expires : strtotime($expires);
$this->headers['Expires'] = gmdate('D, d M Y H:i:s', $expires) . ' GMT';
$this->headers['Cache-Control'] = 'max-age='.($expires - time());
if (isset($this->headers['Pragma']) && $this->headers['Pragma'] == 'no-cache'){
unset($this->headers['Pragma']);
}
14 years ago
}
return $this;
}
/**
* Sends HTTP headers.
*
* @return object Self reference
14 years ago
*/
public function sendHeaders() {
// Send status code header
if (strpos(php_sapi_name(), 'cgi') !== false) {
header(
sprintf(
'Status: %d %s',
$this->status,
self::$codes[$this->status]
),
true
);
}
else {
header(
sprintf(
'%s %d %s',
(isset($_SERVER['SERVER_PROTOCOL']) ? $_SERVER['SERVER_PROTOCOL'] : 'HTTP/1.1'),
$this->status,
self::$codes[$this->status]),
true,
$this->status
);
}
// Send other headers
foreach ($this->headers as $field => $value) {
if (is_array($value)) {
foreach ($value as $v) {
header($field.': '.$v, false);
14 years ago
}
}
else {
header($field.': '.$value);
}
14 years ago
}
// Send content length
$length = $this->getContentLength();
if ($length > 0) {
header('Content-Length: '.$length);
}
return $this;
}
/**
* Gets the content length.
*
* @return string Content length
*/
public function getContentLength() {
return extension_loaded('mbstring') ?
mb_strlen($this->body, 'latin1') :
strlen($this->body);
}
/**
* Gets whether response was sent.
*/
public function sent() {
return $this->sent;
}
/**
* Sends a HTTP response.
*/
public function send() {
if (ob_get_length() > 0) {
ob_end_clean();
}
if (!headers_sent()) {
$this->sendHeaders();
}
echo $this->body;
$this->sent = true;
14 years ago
}
}