2007-08-17 03:32:14 +00:00
< ? php
/**
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
* Represents a response returned by a controller .
2008-02-25 02:10:37 +00:00
*
* @ package sapphire
* @ subpackage control
2007-08-17 03:32:14 +00:00
*/
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
class SS_HTTPResponse {
2010-11-23 19:38:15 +00:00
/**
* @ var array
*/
2007-08-17 03:32:14 +00:00
protected static $status_codes = 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' ,
301 => 'Moved Permanently' ,
302 => 'Found' ,
303 => 'See Other' ,
304 => 'Not Modified' ,
305 => 'Use Proxy' ,
307 => 'Temporary Redirect' ,
400 => 'Bad Request' ,
401 => 'Unauthorized' ,
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 => 'Request Entity Too Large' ,
414 => 'Request-URI Too Long' ,
415 => 'Unsupported Media Type' ,
416 => 'Request Range Not Satisfiable' ,
417 => 'Expectation Failed' ,
2010-10-14 23:56:33 +00:00
422 => 'Unprocessable Entity' ,
2007-08-17 03:32:14 +00:00
500 => 'Internal Server Error' ,
501 => 'Not Implemented' ,
502 => 'Bad Gateway' ,
503 => 'Service Unavailable' ,
504 => 'Gateway Timeout' ,
505 => 'HTTP Version Not Supported' ,
);
2010-11-23 19:38:15 +00:00
/**
* @ var array
*/
2008-04-22 01:45:55 +00:00
protected static $redirect_codes = array (
301 ,
302 ,
303 ,
304 ,
305 ,
2008-11-28 05:29:52 +00:00
307
2008-04-22 01:45:55 +00:00
);
2010-11-23 19:38:15 +00:00
/**
* @ var Int
*/
2007-08-17 03:32:14 +00:00
protected $statusCode = 200 ;
2010-11-23 19:38:15 +00:00
/**
* @ var String
*/
2008-09-12 01:47:39 +00:00
protected $statusDescription = " OK " ;
2008-08-09 07:03:24 +00:00
/**
* HTTP Headers like " Content-Type: text/xml "
*
* @ see http :// en . wikipedia . org / wiki / List_of_HTTP_headers
* @ var array
*/
2009-10-06 21:44:38 +00:00
protected $headers = array (
" Content-Type " => " text/html; charset= \" utf-8 \" " ,
);
2008-08-09 07:03:24 +00:00
/**
* @ var string
*/
2007-08-17 03:32:14 +00:00
protected $body = null ;
2008-09-12 01:47:39 +00:00
/**
* Create a new HTTP response
2010-11-23 19:38:15 +00:00
*
2008-09-12 01:47:39 +00:00
* @ param $body The body of the response
* @ param $statusCode The numeric status code - 200 , 404 , etc
2010-11-23 19:38:15 +00:00
* @ param $statusDescription The text to be given alongside the status code .
* See { @ link setStatusCode ()} for more information .
2008-09-12 01:47:39 +00:00
*/
function __construct ( $body = null , $statusCode = null , $statusDescription = null ) {
$this -> body = $body ;
if ( $statusCode ) $this -> setStatusCode ( $statusCode , $statusDescription );
}
2010-11-23 19:38:15 +00:00
/**
* @ param String $code
* @ param String $description Optional . See { @ link setStatusDescription ()} .
* No newlines are allowed in the description .
* If omitted , will default to the standard HTTP description
* for the given $code value ( see { @ link $status_codes }) .
*/
2008-09-12 01:47:39 +00:00
function setStatusCode ( $code , $description = null ) {
2007-08-17 03:32:14 +00:00
if ( isset ( self :: $status_codes [ $code ])) $this -> statusCode = $code ;
else user_error ( " Unrecognised HTTP status code ' $code ' " , E_USER_WARNING );
2008-09-12 01:47:39 +00:00
if ( $description ) $this -> statusDescription = $description ;
else $this -> statusDescription = self :: $status_codes [ $code ];
2007-08-17 03:32:14 +00:00
}
2008-08-09 07:03:24 +00:00
2010-11-23 19:38:15 +00:00
/**
* The text to be given alongside the status code ( " reason phrase " ) .
* Caution : Will be overwritten by { @ link setStatusCode ()} .
*
* @ param String $description
*/
function setStatusDescription ( $description ) {
$this -> statusDescription = $description ;
}
/**
* @ return Int
*/
2007-12-02 21:29:31 +00:00
function getStatusCode () {
return $this -> statusCode ;
}
2008-08-11 03:03:52 +00:00
/**
* @ return string Description for a HTTP status code
*/
function getStatusDescription () {
2010-11-23 19:38:15 +00:00
return str_replace ( array ( " \r " , " \n " ), '' , $this -> statusDescription );
2008-08-11 03:03:52 +00:00
}
2007-08-17 03:32:14 +00:00
2008-08-11 00:03:57 +00:00
/**
* Returns true if this HTTP response is in error
*/
function isError () {
return $this -> statusCode && ( $this -> statusCode < 200 || $this -> statusCode > 399 );
}
2007-08-17 03:32:14 +00:00
function setBody ( $body ) {
$this -> body = $body ;
}
2008-08-09 07:03:24 +00:00
2007-08-17 03:32:14 +00:00
function getBody () {
return $this -> body ;
}
2007-08-27 05:09:54 +00:00
/**
2008-08-09 07:03:24 +00:00
* Add a HTTP header to the response , replacing any header of the same name .
*
* @ param string $header Example : " Content-Type "
* @ param string $value Example : " text/xml "
2007-08-27 05:09:54 +00:00
*/
2007-08-17 03:32:14 +00:00
function addHeader ( $header , $value ) {
$this -> headers [ $header ] = $value ;
}
2007-08-27 05:09:54 +00:00
/**
2008-08-09 07:03:24 +00:00
* Return the HTTP header of the given name .
*
* @ param string $header
2007-08-27 05:09:54 +00:00
* @ returns string
*/
function getHeader ( $header ) {
2007-08-31 00:26:41 +00:00
if ( isset ( $this -> headers [ $header ])) {
return $this -> headers [ $header ];
} else {
return null ;
}
2007-08-27 05:09:54 +00:00
}
2008-08-09 07:03:24 +00:00
/**
* @ return array
*/
function getHeaders () {
return $this -> headers ;
}
/**
* Remove an existing HTTP header by its name ,
* e . g . " Content-Type " .
*
* @ param unknown_type $header
*/
function removeHeader ( $header ) {
if ( isset ( $this -> headers [ $header ])) unset ( $this -> headers [ $header ]);
}
2008-04-22 01:45:55 +00:00
function redirect ( $dest , $code = 302 ) {
if ( ! in_array ( $code , self :: $redirect_codes )) $code = 302 ;
2011-03-16 22:18:57 +13:00
$this -> setStatusCode ( $code );
2007-08-17 03:32:14 +00:00
$this -> headers [ 'Location' ] = $dest ;
}
/**
* Send this HTTPReponse to the browser
*/
function output () {
2008-11-12 04:31:33 +00:00
// Attach appropriate X-Include-JavaScript and X-Include-CSS headers
if ( Director :: is_ajax ()) {
Requirements :: include_in_response ( $this );
}
2008-11-28 05:29:52 +00:00
2008-04-22 01:45:55 +00:00
if ( in_array ( $this -> statusCode , self :: $redirect_codes ) && headers_sent ( $file , $line )) {
2007-08-17 03:32:14 +00:00
$url = $this -> headers [ 'Location' ];
2008-08-11 03:03:52 +00:00
echo
2007-08-17 03:32:14 +00:00
" <p>Redirecting to <a href= \" $url\ " title = \ " Please click this link if your browser does not redirect you \" > $url ... (output started on $file , line $line )</a></p>
< meta http - equiv = \ " refresh \" content= \" 1; url= $url\ " />
< script type = \ " text/javascript \" >setTimeout('window.location.href = \" $url\ " ' , 50 ); </ script > " ;
} else {
if ( ! headers_sent ()) {
2008-09-29 20:23:47 +00:00
header ( $_SERVER [ 'SERVER_PROTOCOL' ] . " $this->statusCode " . $this -> getStatusDescription ());
2007-08-17 03:32:14 +00:00
foreach ( $this -> headers as $header => $value ) {
header ( " $header : $value " );
}
}
2009-04-03 18:57:17 +00:00
// Only show error pages or generic "friendly" errors if the status code signifies
// an error, and the response doesn't have any body yet that might contain
// a more specific error description.
if ( Director :: isLive () && $this -> isError () && ! $this -> body ) {
2008-10-01 14:43:43 +00:00
Debug :: friendlyError ( $this -> statusCode , $this -> getStatusDescription ());
} else {
echo $this -> body ;
}
2007-08-17 03:32:14 +00:00
}
}
/**
* Returns true if this response is " finished " , that is , no more script execution should be done .
* Specifically , returns true if a redirect has already been requested
*/
function isFinished () {
2009-03-19 23:04:48 +00:00
return in_array ( $this -> statusCode , array ( 301 , 302 , 401 , 403 ));
2007-08-17 03:32:14 +00:00
}
2009-10-11 00:06:53 +00:00
/**
* @ deprecated 2.4 Use { @ link HTTP :: getLinksIn ()} on DOMDocument .
*/
public function getLinks () {
user_error (
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
'SS_HTTPResponse->getLinks() is deprecated, please use HTTP::getLinksIn() or DOMDocument.' , E_USER_NOTICE
2009-10-11 00:06:53 +00:00
);
$attributes = array ( 'id' , 'href' , 'class' );
$links = array ();
$results = array ();
if ( preg_match_all ( '/<a[^>]+>/i' , $this -> body , $links )) foreach ( $links [ 0 ] as $link ) {
2007-08-20 22:38:37 +00:00
$processedLink = array ();
2009-10-11 00:06:53 +00:00
foreach ( $attributes as $attribute ) {
$matches = array ();
if ( preg_match ( '/' . $attribute . '\s*=\s*"([^"]+)"/i' , $link , $matches )) {
$processedLink [ $attribute ] = $matches [ 1 ];
}
}
2007-08-20 22:38:37 +00:00
$results [] = $processedLink ;
2009-10-11 00:06:53 +00:00
}
2007-08-20 22:38:37 +00:00
return $results ;
}
2007-08-17 03:32:14 +00:00
2009-03-10 22:17:26 +00:00
}
2009-06-27 08:48:44 +00:00
/**
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
* A { @ link SS_HTTPResponse } encapsulated in an exception , which can interrupt the processing flow and be caught by the
2009-06-27 08:48:44 +00:00
* { @ link RequestHandler } and returned to the user .
*
* Example Usage :
* < code >
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
* throw new SS_HTTPResponse_Exception ( 'This request was invalid.' , 400 );
* throw new SS_HTTPResponse_Exception ( new SS_HTTPResponse ( 'There was an internal server error.' , 500 ));
2009-06-27 08:48:44 +00:00
* </ code >
*
* @ package sapphire
* @ subpackage control
*/
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
class SS_HTTPResponse_Exception extends Exception {
2009-06-27 08:48:44 +00:00
protected $response ;
/**
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
* @ see SS_HTTPResponse :: __construct ();
2009-06-27 08:48:44 +00:00
*/
public function __construct ( $body = null , $statusCode = null , $statusDescription = null ) {
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
if ( $body instanceof SS_HTTPResponse ) {
2009-06-27 08:48:44 +00:00
$this -> setResponse ( $body );
} else {
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
$this -> setResponse ( new SS_HTTPResponse ( $body , $statusCode , $statusDescription ));
2009-06-27 08:48:44 +00:00
}
parent :: __construct ( $this -> getResponse () -> getBody (), $this -> getResponse () -> getStatusCode ());
}
/**
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
* @ return SS_HTTPResponse
2009-06-27 08:48:44 +00:00
*/
public function getResponse () {
return $this -> response ;
}
/**
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
* @ param SS_HTTPResponse $response
2009-06-27 08:48:44 +00:00
*/
API CHANGE: Renamed conflicting classes to have an "SS_" namespace, and renamed existing "SS" namespace to "SS_". The affected classes are: HTTPRequest, HTTPResponse, Query, Database, SSBacktrace, SSCli, SSDatetime, SSDatetimeTest, SSLog, SSLogTest, SSLogEmailWriter, SSLogErrorEmailFormatter, SSLogErrorFileFormatter, SSLogFileWriter and SSZendLog.
MINOR: Replaced usage of renamed classes with the new namespaced name.
From: Andrew Short <andrewjshort@gmail.com>
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@90075 467b73ca-7a2a-4603-9d3b-597d59a354a9
2009-10-26 03:06:31 +00:00
public function setResponse ( SS_HTTPResponse $response ) {
2009-06-27 08:48:44 +00:00
$this -> response = $response ;
}
}