response.php 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333
  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)
  196. {
  197. $response = new static($response);
  198. }
  199. // We'll need to force the response to be a string before closing the session,
  200. // since the developer may be using the session within a view, and we can't
  201. // age the flash data until the view is rendered.
  202. //
  203. // Since this method is used by both the Route and Controller classes, it is
  204. // a convenient spot to cast the application response to a string before it
  205. // is returned to the main request handler.
  206. $response->render();
  207. return $response;
  208. }
  209. /**
  210. * Convert the content of the Response to a string and return it.
  211. *
  212. * @return string
  213. */
  214. public function render()
  215. {
  216. if (is_object($this->content) and method_exists($this->content, '__toString'))
  217. {
  218. $this->content = $this->content->__toString();
  219. }
  220. else
  221. {
  222. $this->content = (string) $this->content;
  223. }
  224. return $this->content;
  225. }
  226. /**
  227. * Send the headers and content of the response to the browser.
  228. *
  229. * @return void
  230. */
  231. public function send()
  232. {
  233. if ( ! headers_sent()) $this->send_headers();
  234. echo (string) $this->content;
  235. }
  236. /**
  237. * Send all of the response headers to the browser.
  238. *
  239. * @return void
  240. */
  241. public function send_headers()
  242. {
  243. // If the server is using FastCGI, we need to send a slightly different
  244. // protocol and status header than we normally would. Otherwise it will
  245. // not call any custom scripts setup to handle 404 responses.
  246. //
  247. // The status header will contain both the code and the status message,
  248. // such as "OK" or "Not Found". For typical servers, the HTTP protocol
  249. // will also be included with the status.
  250. if (isset($_SERVER['FCGI_SERVER_VERSION']))
  251. {
  252. header('Status: '.$this->status.' '.$this->message());
  253. }
  254. else
  255. {
  256. header(Request::protocol().' '.$this->status.' '.$this->message());
  257. }
  258. // If the content type was not set by the developer, we will set the
  259. // header to a default value that indicates to the browser that the
  260. // response is HTML and that it uses the default encoding.
  261. if ( ! isset($this->headers['Content-Type']))
  262. {
  263. $encoding = Config::get('application.encoding');
  264. $this->header('Content-Type', 'text/html; charset='.$encoding);
  265. }
  266. // Once the framework controlled headers have been sentm, we can
  267. // simply iterate over the developer's headers and send each one
  268. // back to the browser for the response.
  269. foreach ($this->headers as $name => $value)
  270. {
  271. header("{$name}: {$value}", true);
  272. }
  273. }
  274. /**
  275. * Get the status code message for the response.
  276. *
  277. * @return string
  278. */
  279. public function message()
  280. {
  281. return static::$statuses[$this->status];
  282. }
  283. /**
  284. * Add a header to the array of response headers.
  285. *
  286. * @param string $name
  287. * @param string $value
  288. * @return Response
  289. */
  290. public function header($name, $value)
  291. {
  292. $this->headers[$name] = $value;
  293. return $this;
  294. }
  295. /**
  296. * Set the response status code.
  297. *
  298. * @param int $status
  299. * @return Response
  300. */
  301. public function status($status)
  302. {
  303. $this->status = $status;
  304. return $this;
  305. }
  306. }