response.php 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330
  1. <?php namespace Laravel;
  2. class Response {
  3. /**
  4. * The content of the response.
  5. *
  6. * @var mixed
  7. */
  8. public $content;
  9. /**
  10. * The HTTP status code of the response.
  11. *
  12. * @var int
  13. */
  14. public $status = 200;
  15. /**
  16. * The response headers.
  17. *
  18. * @var array
  19. */
  20. public $headers = array();
  21. /**
  22. * HTTP status codes.
  23. *
  24. * @var array
  25. */
  26. public static $statuses = array(
  27. 100 => 'Continue',
  28. 101 => 'Switching Protocols',
  29. 200 => 'OK',
  30. 201 => 'Created',
  31. 202 => 'Accepted',
  32. 203 => 'Non-Authoritative Information',
  33. 204 => 'No Content',
  34. 205 => 'Reset Content',
  35. 206 => 'Partial Content',
  36. 207 => 'Multi-Status',
  37. 300 => 'Multiple Choices',
  38. 301 => 'Moved Permanently',
  39. 302 => 'Found',
  40. 303 => 'See Other',
  41. 304 => 'Not Modified',
  42. 305 => 'Use Proxy',
  43. 307 => 'Temporary Redirect',
  44. 400 => 'Bad Request',
  45. 401 => 'Unauthorized',
  46. 402 => 'Payment Required',
  47. 403 => 'Forbidden',
  48. 404 => 'Not Found',
  49. 405 => 'Method Not Allowed',
  50. 406 => 'Not Acceptable',
  51. 407 => 'Proxy Authentication Required',
  52. 408 => 'Request Timeout',
  53. 409 => 'Conflict',
  54. 410 => 'Gone',
  55. 411 => 'Length Required',
  56. 412 => 'Precondition Failed',
  57. 413 => 'Request Entity Too Large',
  58. 414 => 'Request-URI Too Long',
  59. 415 => 'Unsupported Media Type',
  60. 416 => 'Requested Range Not Satisfiable',
  61. 417 => 'Expectation Failed',
  62. 422 => 'Unprocessable Entity',
  63. 423 => 'Locked',
  64. 424 => 'Failed Dependency',
  65. 500 => 'Internal Server Error',
  66. 501 => 'Not Implemented',
  67. 502 => 'Bad Gateway',
  68. 503 => 'Service Unavailable',
  69. 504 => 'Gateway Timeout',
  70. 505 => 'HTTP Version Not Supported',
  71. 507 => 'Insufficient Storage',
  72. 509 => 'Bandwidth Limit Exceeded'
  73. );
  74. /**
  75. * Create a new response instance.
  76. *
  77. * @param mixed $content
  78. * @param int $status
  79. * @param array $headers
  80. * @return void
  81. */
  82. public function __construct($content, $status = 200, $headers = array())
  83. {
  84. $this->status = $status;
  85. $this->content = $content;
  86. $this->headers = $headers;
  87. }
  88. /**
  89. * Create a new response instance.
  90. *
  91. * <code>
  92. * // Create a response instance with string content
  93. * return Response::make(json_encode($user));
  94. *
  95. * // Create a response instance with a given status
  96. * return Response::make('Not Found', 404);
  97. *
  98. * // Create a response with some custom headers
  99. * return Response::make(json_encode($user), 200, array('header' => 'value'));
  100. * </code>
  101. *
  102. * @param mixed $content
  103. * @param int $status
  104. * @param array $headers
  105. * @return Response
  106. */
  107. public static function make($content, $status = 200, $headers = array())
  108. {
  109. return new static($content, $status, $headers);
  110. }
  111. /**
  112. * Create a new response instance containing a view.
  113. *
  114. * <code>
  115. * // Create a response instance with a view
  116. * return Response::view('home.index');
  117. *
  118. * // Create a response instance with a view and data
  119. * return Response::view('home.index', array('name' => 'Taylor'));
  120. * </code>
  121. *
  122. * @param string $view
  123. * @param array $data
  124. * @return Response
  125. */
  126. public static function view($view, $data = array())
  127. {
  128. return new static(View::make($view, $data));
  129. }
  130. /**
  131. * Create a new error response instance.
  132. *
  133. * The response status code will be set using the specified code.
  134. *
  135. * The specified error should match a view in your views/error directory.
  136. *
  137. * <code>
  138. * // Create a 404 response
  139. * return Response::error('404');
  140. *
  141. * // Create a 404 response with data
  142. * return Response::error('404', array('message' => 'Not Found'));
  143. * </code>
  144. *
  145. * @param int $code
  146. * @param array $data
  147. * @return Response
  148. */
  149. public static function error($code, $data = array())
  150. {
  151. return new static(View::make('error.'.$code, $data), $code);
  152. }
  153. /**
  154. * Create a new download response instance.
  155. *
  156. * <code>
  157. * // Create a download response to a given file
  158. * return Response::download('path/to/file.jpg');
  159. *
  160. * // Create a download response with a given file name
  161. * return Response::download('path/to/file.jpg', 'your_file.jpg');
  162. * </code>
  163. *
  164. * @param string $path
  165. * @param string $name
  166. * @param array $headers
  167. * @return Response
  168. */
  169. public static function download($path, $name = null, $headers = array())
  170. {
  171. if (is_null($name)) $name = basename($path);
  172. $headers = array_merge(array(
  173. 'Content-Description' => 'File Transfer',
  174. 'Content-Type' => File::mime(File::extension($path)),
  175. 'Content-Disposition' => 'attachment; filename="'.$name.'"',
  176. 'Content-Transfer-Encoding' => 'binary',
  177. 'Expires' => 0,
  178. 'Cache-Control' => 'must-revalidate, post-check=0, pre-check=0',
  179. 'Pragma' => 'public',
  180. 'Content-Length' => File::size($path),
  181. ), $headers);
  182. return new static(File::get($path), 200, $headers);
  183. }
  184. /**
  185. * Prepare a response from the given value.
  186. *
  187. * If the value is not a response, it will be converted into a response
  188. * instance and the content will be cast to a string.
  189. *
  190. * @param mixed $response
  191. * @return Response
  192. */
  193. public static function prepare($response)
  194. {
  195. if ( ! $response instanceof Response) $response = new static($response);
  196. // We'll need to force the response to be a string before closing the session,
  197. // since the developer may be using the session within a view, and we can't
  198. // age the flash data until the view is rendered.
  199. //
  200. // Since this method is used by both the Route and Controller classes, it is
  201. // a convenient spot to cast the application response to a string before it
  202. // is returned to the main request handler.
  203. $response->render();
  204. return $response;
  205. }
  206. /**
  207. * Convert the content of the Response to a string and return it.
  208. *
  209. * @return string
  210. */
  211. public function render()
  212. {
  213. if (is_object($this->content) and method_exists($this->content, '__toString'))
  214. {
  215. $this->content = $this->content->__toString();
  216. }
  217. else
  218. {
  219. $this->content = (string) $this->content;
  220. }
  221. return $this->content;
  222. }
  223. /**
  224. * Send the headers and content of the response to the browser.
  225. *
  226. * @return void
  227. */
  228. public function send()
  229. {
  230. if ( ! headers_sent()) $this->send_headers();
  231. echo (string) $this->content;
  232. }
  233. /**
  234. * Send all of the response headers to the browser.
  235. *
  236. * @return void
  237. */
  238. public function send_headers()
  239. {
  240. // If the server is using FastCGI, we need to send a slightly different
  241. // protocol and status header than we normally would. Otherwise it will
  242. // not call any custom scripts setup to handle 404 responses.
  243. //
  244. // The status header will contain both the code and the status message,
  245. // such as "OK" or "Not Found". For typical servers, the HTTP protocol
  246. // will also be included with the status.
  247. if (isset($_SERVER['FCGI_SERVER_VERSION']))
  248. {
  249. header('Status: '.$this->status.' '.$this->message());
  250. }
  251. else
  252. {
  253. header(Request::protocol().' '.$this->status.' '.$this->message());
  254. }
  255. // If the content type was not set by the developer, we will set the
  256. // header to a default value that indicates to the browser that the
  257. // response is HTML and that it uses the default encoding.
  258. if ( ! isset($this->headers['Content-Type']))
  259. {
  260. $encoding = Config::get('application.encoding');
  261. $this->header('Content-Type', 'text/html; charset='.$encoding);
  262. }
  263. // Once the framework controlled headers have been sentm, we can
  264. // simply iterate over the developer's headers and send each one
  265. // back to the browser for the response.
  266. foreach ($this->headers as $name => $value)
  267. {
  268. header("{$name}: {$value}", true);
  269. }
  270. }
  271. /**
  272. * Get the status code message for the response.
  273. *
  274. * @return string
  275. */
  276. public function message()
  277. {
  278. return static::$statuses[$this->status];
  279. }
  280. /**
  281. * Add a header to the array of response headers.
  282. *
  283. * @param string $name
  284. * @param string $value
  285. * @return Response
  286. */
  287. public function header($name, $value)
  288. {
  289. $this->headers[$name] = $value;
  290. return $this;
  291. }
  292. /**
  293. * Set the response status code.
  294. *
  295. * @param int $status
  296. * @return Response
  297. */
  298. public function status($status)
  299. {
  300. $this->status = $status;
  301. return $this;
  302. }
  303. }