[Top][All Lists]
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]
[Phpgroupware-cvs] phpgwapi/inc/class.net_http_client.inc.php, 1.3
From: |
nomail |
Subject: |
[Phpgroupware-cvs] phpgwapi/inc/class.net_http_client.inc.php, 1.3 |
Date: |
Thu, 30 Dec 2004 07:47:31 +0100 |
Update of /phpgwapi/inc
Added Files:
Branch:
class.net_http_client.inc.php
date: 2004/12/30 06:47:31; author: skwashd; state: Exp; lines: +750 -800
Log Message:
new HEAD
=====================================================================
<?php
/**
* HTTP and WebDAV client protocol class
* @author Leo West <address@hidden>
* @copyright Copyright (C) 2001,2002 Leo West
* @copyright Portions Copyright (C) 2003,2004 Free Software Foundation,
Inc. http://www.fsf.org/
* @license http://www.fsf.org/licenses/gpl.html GNU General Public
License
* @package phpgwapi
* @subpackage network
* @link http://lwest.free.fr/doc/php/lib/net_http_client-en.html
* @internal Version 0.7
* @todo remaining WebDAV methods: UNLOCK PROPPATCH
*/
/**
* Debug level - debug methods calls
*/
define( "DBGTRACE", 1 );
/**
* Debug level - debug data received
*/
define( "DBGINDATA", 2 );
/**
* Debug level - debug data sent
*/
define( "DBGOUTDATA", 4 );
/**
* Debug level - debug low-level (usually internal) methods
*/
define( "DBGLOW", 8 );
/**
* Debug level - debug socket-level code
*/
define( "DBGSOCK", 16 );
/**
* Internal error: connection failed
*/
define( "ECONNECTION", -1 );
/**
* Internal error: response status line is not http compliant
*/
define( "EBADRESPONSE", -2 );
/**
* CR/LF
*/
define( "CRLF", "\r\n" );
/**
* HTTP and WebDAV client protocol class
*
* @package phpgwapi
* @subpackage network
*/
class net_http_client
{
// @private
/// array containg server URL, similar to array returned by
parseurl()
var $url;
/// server response code eg. "304"
var $reply;
/// server response line eg. "200 OK"
var $replyString;
/// HTPP protocol version used
var $protocolVersion = "1.0";
/// internal buffers
var $requestHeaders, $requestBody;
/// TCP socket identifier
var $socket = false;
/// proxy informations
var $useProxy = false;
var $proxyHost, $proxyPort;
/// debugging flag
var $debug = 0;
/**
* net_http_client
* constructor
* Note : when host and port are defined, the connection is
immediate
* @see Connect()
**/
function net_http_client( $host= NULL, $port= NULL )
{
if( $this->debug & DBGTRACE ) echo "net_http_client(
$host, $port )\n";
if( $host != NULL ) {
$this->connect( $host, $port );
}
}
/**
* turn on debug messages
* @param level a combinaison of debug flags
* @see debug flags ( DBG..) defined at top of file
**/
function setDebug( $level )
{
if( $this->debug & DBGTRACE ) echo "setDebug( $level
)\n";
$this->debug = $level;
}
/**
* turn on proxy support
* @param proxyHost proxy host address eg "proxy.mycorp.com"
* @param proxyPort proxy port usually 80 or 8080
**/
function setProxy( $proxyHost, $proxyPort )
{
if( $this->debug & DBGTRACE ) echo "setProxy(
$proxyHost, $proxyPort )\n";
$this->useProxy = true;
$this->proxyHost = $proxyHost;
$this->proxyPort = $proxyPort;
}
/**
* setProtocolVersion
* define the HTTP protocol version to use
* @param version string the version number with one decimal:
"0.9", "1.0", "1.1"
* when using 1.1, you MUST set the mandatory headers "Host"
* @return boolean false if the version number is bad, true if
ok
**/
function setProtocolVersion( $version )
{
if( $this->debug & DBGTRACE ) echo "setProtocolVersion(
$version )\n";
if( $version > 0 and $version <= 1.1 ) {
$this->protocolVersion = $version;
return true;
} else {
return false;
}
}
/**
* set a username and password to access a protected resource
* Only "Basic" authentication scheme is supported yet
* @param username string - identifier
* @param password string - clear password
**/
function setCredentials( $username, $password )
{
$hdrvalue = base64_encode( "$username:$password" );
$this->addHeader( "Authorization", "Basic $hdrvalue" );
}
/**
* define a set of HTTP headers to be sent to the server
* header names are lowercased to avoid duplicated headers
* @param headers hash array containing the headers as
headerName => headerValue pairs
**/
function setHeaders( $headers )
{
if( $this->debug & DBGTRACE ) echo "setHeaders(
$headers ) \n";
if( is_array( $headers )) {
foreach( $headers as $name => $value ) {
$this->requestHeaders[$name] = $value;
}
}
}
/**
* addHeader
* set a unique request header
* @param headerName the header name
* @param headerValue the header value, ( unencoded)
**/
function addHeader( $headerName, $headerValue )
{
if( $this->debug & DBGTRACE ) echo "addHeader(
$headerName, $headerValue )\n";
$this->requestHeaders[$headerName] = $headerValue;
}
/**
* removeHeader
* unset a request header
* @param headerName the header name
**/
function removeHeader( $headerName )
{
if( $this->debug & DBGTRACE ) echo "removeHeader(
$headerName) \n";
unset( $this->requestHeaders[$headerName] );
}
/**
* addCookie
* set a session cookie, that will be used in the next requests.
* this is a hack as cookie are usually set by the server, but
you may need it
* it is your responsabilty to unset the cookie if you request
another host
* to keep a session on the server
* @param string the name of the cookie
* @param string the value for the cookie
**/
function addCookie( $cookiename, $cookievalue )
{
if( $this->debug & DBGTRACE ) echo "addCookie(
$cookiename, $cookievalue ) \n";
$cookie = $cookiename . "=" . $cookievalue;
$this->requestHeaders["Cookie"] = $cookie;
}
/**
* removeCookie
* unset cookies currently in use
**/
function removeCookies()
{
if( $this->debug & DBGTRACE ) echo "removeCookies()
\n";
unset( $this->requestHeaders["Cookie"] );
}
/**
* Connect
* open the connection to the server
* @param host string server address (or IP)
* @param port string server listening port - defaults to 80
* @return boolean false is connection failed, true otherwise
**/
function Connect( $host, $port = NULL )
{
if( $this->debug & DBGTRACE ) echo "Connect( $host,
$port ) \n";
$this->url['scheme'] = "http";
$this->url['host'] = $host;
if( $port != NULL )
$this->url['port'] = $port;
return true;
}
/**
* Disconnect
* close the connection to the server
**/
function Disconnect()
{
if( $this->debug & DBGTRACE ) echo "Disconnect()\n";
if( $this->socket )
fclose( $this->socket );
}
/**
* head
* issue a HEAD request
* @param uri string URI of the document
* @return string response status code (200 if ok)
* @see getHeaders()
**/
function Head( $uri )
{
if( $this->debug & DBGTRACE ) echo "Head( $uri )\n";
$this->responseHeaders = $this->responseBody = "";
$uri = $this->makeUri( $uri );
if( $this->sendCommand( "HEAD $uri
HTTP/$this->protocolVersion" ) )
$this->processReply();
return $this->reply;
}
/**
* get
* issue a GET http request
* @param uri URI (path on server) or full URL of the document
* @return string response status code (200 if ok)
* @see getHeaders(), getBody()
**/
function Get( $url )
{
if( $this->debug & DBGTRACE ) echo "Get( $url )\n";
$this->responseHeaders = $this->responseBody = "";
$uri = $this->makeUri( $url );
if( $this->sendCommand( "GET $uri
HTTP/$this->protocolVersion" ) )
$this->processReply();
return $this->reply;
}
/**
* Options
* issue a OPTIONS http request
* @param uri URI (path on server) or full URL of the document
* @return array list of options supported by the server or
NULL in case of error
**/
function Options( $url )
{
if( $this->debug & DBGTRACE ) echo "Options( $url )\n";
$this->responseHeaders = $this->responseBody = "";
$uri = $this->makeUri( $url );
if( $this->sendCommand( "OPTIONS $uri
HTTP/$this->protocolVersion" ) )
$this->processReply();
if( @$this->responseHeaders["Allow"] == NULL )
return NULL;
else
return explode( ",",
$this->responseHeaders["Allow"] );
}
/**
* Post
* issue a POST http request
* @param uri string URI of the document
* @param query_params array parameters to send in the form
"parameter name" => value
* @return string response status code (200 if ok)
*
* $params = array( "login" => "tiger", "password" => "secret"
);
* $http->post( "/login.php", $params );
**/
function Post( $uri, $query_params="" )
{
if( $this->debug & DBGTRACE ) echo "Post( $uri,
$query_params )\n";
$uri = $this->makeUri( $uri );
if( is_array($query_params) ) {
$postArray = array();
foreach( $query_params as $k=>$v ) {
$postArray[] = urlencode($k) . "=" .
urlencode($v);
}
$this->requestBody = implode( "&", $postArray);
}
// set the content type for post parameters
$this->addHeader( 'Content-Type',
"application/x-www-form-urlencoded" );
// done in sendCommand() $this->addHeader(
'Content-Length', strlen($this->requestBody) );
if( $this->sendCommand( "POST $uri
HTTP/$this->protocolVersion" ) )
$this->processReply();
$this->removeHeader('Content-Type');
$this->removeHeader('Content-Length');
$this->requestBody = "";
return $this->reply;
}
/**
* Put
* Send a PUT request
* PUT is the method to sending a file on the server. it is
*not* widely supported
* @param uri the location of the file on the server. dont
forget the heading "/"
* @param filecontent the content of the file. binary content
accepted
* @return string response status code 201 (Created) if ok
* @see RFC2518 "HTTP Extensions for Distributed Authoring
WEBDAV"
**/
function Put( $uri, $filecontent )
{
if( $this->debug & DBGTRACE ) echo "Put( $uri,
[filecontent not displayed )\n";
$uri = $this->makeUri( $uri );
$this->requestBody = $filecontent;
if( $this->sendCommand( "PUT $uri
HTTP/$this->protocolVersion" ) )
$this->processReply();
return $this->reply;
}
/**
* Send a MOVE HTTP-DAV request
* Move (rename) a file on the server
* @param srcUri the current file location on the server. dont
forget the heading "/"
* @param destUri the destination location on the server. this
is *not* a full URL
* @param overwrite boolean - true to overwrite an existing
destinationn default if yes
* @return string response status code 204 (Unchanged) if ok
* @see RFC2518 "HTTP Extensions for Distributed Authoring
WEBDAV"
**/
function Move( $srcUri, $destUri, $overwrite=true, $scope=0 )
{
if( $this->debug & DBGTRACE ) echo "Move( $srcUri,
$destUri, $overwrite )\n";
if( $overwrite )
$this->requestHeaders['Overwrite'] = "T";
else
$this->requestHeaders['Overwrite'] = "F";
/*
$destUrl = $this->url['scheme'] . "://" .
$this->url['host'];
if( $this->url['port'] != "" )
$destUrl .= ":" . $this->url['port'];
$destUrl .= $destUri;
$this->requestHeaders['Destination'] = $destUrl;
*/
$this->requestHeaders['Destination'] = $destUri;
$this->requestHeaders['Depth']=$scope;
if( $this->sendCommand( "MOVE $srcUri
HTTP/$this->protocolVersion" ) )
$this->processReply();
return $this->reply;
}
/**
* Send a COPY HTTP-DAV request
* Copy a file -allready on the server- into a new location
* @param srcUri the current file location on the server. dont
forget the heading "/"
* @param destUri the destination location on the server. this
is *not* a full URL
* @param overwrite boolean - true to overwrite an existing
destination - overwrite by default
* @return string response status code 204 (Unchanged) if ok
* @see RFC2518 "HTTP Extensions for Distributed Authoring
WEBDAV"
**/
function Copy( $srcUri, $destUri, $overwrite=true, $scope=0)
{
if( $this->debug & DBGTRACE ) echo "Copy( $srcUri,
$destUri, $overwrite )\n";
if( $overwrite )
$this->requestHeaders['Overwrite'] = "T";
else
$this->requestHeaders['Overwrite'] = "F";
/*
$destUrl = $this->url['scheme'] . "://" .
$this->url['host'];
if( $this->url['port'] != "" )
$destUrl .= ":" . $this->url['port'];
$destUrl .= $destUri;
$this->requestHeaders['Destination'] = $destUrl;
*/
$this->requestHeaders['Destination'] = $destUri;
$this->requestHeaders['Depth']=$scope;
if( $this->sendCommand( "COPY $srcUri
HTTP/$this->protocolVersion" ) )
$this->processReply();
return $this->reply;
}
/**
* Send a MKCOL HTTP-DAV request
* Create a collection (directory) on the server
* @param uri the directory location on the server. dont forget
the heading "/"
* @return string response status code 201 (Created) if ok
* @see RFC2518 "HTTP Extensions for Distributed Authoring
WEBDAV"
**/
function MkCol( $uri )
{
if( $this->debug & DBGTRACE ) echo "Mkcol( $uri )\n";
// $this->requestHeaders['Overwrite'] = "F";
$this->requestHeaders['Depth']=0;
if( $this->sendCommand( "MKCOL $uri
HTTP/$this->protocolVersion" ) )
$this->processReply();
return $this->reply;
}
/**
* Delete a file on the server using the "DELETE" HTTP-DAV
request
* This HTTP method is *not* widely supported
* Only partially supports "collection" deletion, as the XML
response is not parsed
* @param uri the location of the file on the server. dont
forget the heading "/"
* @return string response status code 204 (Unchanged) if ok
* @see RFC2518 "HTTP Extensions for Distributed Authoring
WEBDAV"
**/
function Delete( $uri, $scope=0)
{
if( $this->debug & DBGTRACE ) echo "Delete( $uri )\n";
$this->requestHeaders['Depth'] = $scope;
if( $this->sendCommand( "DELETE $uri
HTTP/$this->protocolVersion" ) ){
$this->processReply();
}
return $this->reply;
}
/**
* PropFind
* implements the PROPFIND method
* PROPFIND retrieves meta informations about a resource on the
server
* XML reply is not parsed, you'll need to do it
* @param uri the location of the file on the server. dont
forget the heading "/"
* @param scope set the scope of the request.
* O : infos about the node only
* 1 : infos for the node and its direct children ( one
level)
* Infinity : infos for the node and all its children
nodes (recursive)
* @return string response status code - 207 (Multi-Status) if
OK
* @see RFC2518 "HTTP Extensions for Distributed Authoring
WEBDAV"
**/
function PropFind( $uri, $scope=0 )
{
$this->requestBody = '';
if( $this->debug & DBGTRACE ) echo "Propfind( $uri,
$scope )\n";
$prev_depth=$this->requestHeaders['Depth'];
$this->requestHeaders['Depth'] = $scope;
if( $this->sendCommand( "PROPFIND $uri
HTTP/$this->protocolVersion" ) )
$this->processReply();
$this->requestHeaders['Depth']=$prev_depth;
return $this->reply;
}
/**
* Lock - WARNING: EXPERIMENTAL
* Lock a ressource on the server. XML reply is not parsed,
you'll need to do it
* @param $uri URL (relative) of the resource to lock
* @param $lockScope - use "exclusive" for an eclusive lock,
"inclusive" for a shared lock
* @param $lockType - acces type of the lock : "write"
* @param $lockScope - use "exclusive" for an eclusive lock,
"inclusive" for a shared lock
* @param $lockOwner - an url representing the owner for this
lock
* @return server reply code, 200 if ok
**/
function Lock( $uri, $lockScope, $lockType, $lockOwner )
{
$body = "<?xml version=\"1.0\" encoding=\"utf-8\" ?>
<D:lockinfo xmlns:D='DAV:'>
<D:lockscope><D:$lockScope/></D:lockscope>\n<D:locktype><D:$lockType/></D:locktype>
<D:owner><D:href>$lockOwner</D:href></D:owner>
</D:lockinfo>\n";
$this->requestBody = utf8_encode( $body );
if( $this->sendCommand( "LOCK $uri
HTTP/$this->protocolVersion" ) )
$this->processReply();
return $this->reply;
}
/**
* Unlock - WARNING: EXPERIMENTAL
* unlock a ressource on the server
* @param $uri URL (relative) of the resource to unlock
* @param $lockToken the lock token given at lock time, eg:
opaquelocktoken:e71d4fae-5dec-22d6-fea5-00a0c91e6be4
* @return server reply code, 204 if ok
**/
function Unlock( $uri, $lockToken )
{
$this->addHeader( "Lock-Token", "<$lockToken>" );
if( $this->sendCommand( "UNLOCK $uri
HTTP/$this->protocolVersion" ) )
$this->processReply();
return $this->reply;
}
/**
* getHeaders
* return the response headers
* to be called after a Get() or Head() call
* @return array headers received from server in the form
headername => value
* @see Get(), Head()
**/
function getHeaders()
{
if( $this->debug & DBGTRACE ) echo "getHeaders()\n";
if( $this->debug & DBGINDATA ) {
echo "DBG.INDATA responseHeaders="; print_r(
$this->responseHeaders );
}
return $this->responseHeaders;
}
/**
* getHeader
* return the response header "headername"
* @param headername the name of the header
* @return header value or NULL if no such header is defined
**/
function getHeader( $headername )
{
if( $this->debug & DBGTRACE ) echo "getHeaderName(
$headername )\n";
return $this->responseHeaders[$headername];
}
/**
* getBody
* return the response body
* invoke it after a Get() call for instance, to retrieve the
response
* @return string body content
* @see Get(), Head()
**/
function getBody()
{
if( $this->debug & DBGTRACE ) echo "getBody()\n";
return $this->responseBody;
}
/**
* getStatus return the server response's status code
* @return string a status code
* code are divided in classes (where x is a digit)
* - 20x : request processed OK
* - 30x : document moved
* - 40x : client error ( bad url, document not found, etc...)
* - 50x : server error
* @see RFC2616 "Hypertext Transfer Protocol -- HTTP/1.1"
**/
function getStatus()
{
return $this->reply;
}
/**
* getStatusMessage return the full response status, of the
form "CODE Message"
* eg. "404 Document not found"
* @return string the message
**/
function getStatusMessage()
{
return $this->replyString;
}
/**
* send a request
* data sent are in order
* a) the command
* b) the request headers if they are defined
* c) the request body if defined
* @return string the server repsonse status code
* @access protected
**/
function sendCommand( $command )
{
if( $this->debug & DBGLOW ) echo "sendCommand( $command
)\n";
$this->responseHeaders = array();
$this->responseBody = "";
// connect if necessary
if( $this->socket == false or feof( $this->socket) ) {
if( $this->useProxy ) {
$host = $this->proxyHost;
$port = $this->proxyPort;
} else {
$host = $this->url['host'];
$port = $this->url['port'];
}
if( $port == "" ) $port = 80;
$this->socket = fsockopen( $host, $port,
&$this->reply, &$this->replyString );
if( $this->debug & DBGSOCK ) echo "connexion(
$host, $port) - $this->socket\n";
if( ! $this->socket ) {
if( $this->debug & DBGSOCK ) echo
"FAILED : $this->replyString ($this->reply)\n";
return false;
}
}
if( $this->requestBody != "" ) {
$this->addHeader( "Content-Length", strlen(
$this->requestBody ) );
}
else {
$this->removeHeader( "Content-Length");
}
$this->request = $command;
$cmd = $command . CRLF;
if( is_array( $this->requestHeaders) ) {
foreach( $this->requestHeaders as $k => $v ) {
$cmd .= "$k: $v" . CRLF;
}
}
if( $this->requestBody != "" ) {
$cmd .= CRLF . $this->requestBody;
}
// unset body (in case of successive requests)
$this->requestBody = "";
if( $this->debug & DBGOUTDATA ) echo "DBG.OUTDATA
Sending\n$cmd\n";
fputs( $this->socket, $cmd . CRLF );
return true;
}
/**
*
* @access protected
*/
function processReply()
{
if( $this->debug & DBGLOW ) echo "processReply()\n";
$this->replyString = trim(fgets( $this->socket,1024) );
if( preg_match( "|^HTTP/\S+ (\d+) |i",
$this->replyString, $a )) {
$this->reply = $a[1];
} else {
$this->reply = EBADRESPONSE;
}
if( $this->debug & DBGINDATA ) echo "replyLine:
$this->replyString\n";
// get response headers and body
$this->responseHeaders = $this->processHeader();
$this->responseBody = $this->processBody();
if ($this->responseHeaders['Connection'] == 'close') {
if( $this->debug & DBGINDATA ) echo "connection
closed at server request!";
fclose($this->socket);
$this->socket=false;
}
// if( $this->responseHeaders['set-cookie'] )
// $this->addHeader( "cookie",
$this->responseHeaders['set-cookie'] );
return $this->reply;
}
/**
* processHeader() reads header lines from socket until the line
equals $lastLine
* @access protected
* @return array of headers with header names as keys and header
content as values
*/
function processHeader( $lastLine = CRLF )
{
if( $this->debug & DBGLOW ) echo "processHeader(
[lastLine] )\n";
$headers = array();
$finished = false;
while ( ( ! $finished ) && ( ! feof($this->socket)) ) {
$str = fgets( $this->socket, 1024 );
if( $this->debug & DBGINDATA ) echo "HEADER :
$str;";
$finished = ( $str == $lastLine );
if ( !$finished ) {
list( $hdr, $value ) = split( ": ",
$str, 2 );
// nasty workaround broken multiple
same headers (eg. Set-Cookie headers) @FIXME
if( isset( $headers[$hdr]) )
$headers[$hdr] .= "; " .
trim($value);
else
$headers[$hdr] = trim($value);
}
}
return $headers;
}
/**
* processBody() reads the body from the socket
* the body is the "real" content of the reply
* @return string body content
* @access protected
*/
function processBody()
{
$failureCount = 0;
$data='';
if( $this->debug & DBGLOW ) echo "processBody()\n";
if (
$this->responseHeaders['Transfer-Encoding']=='chunked' )
{
// chunked encoding
if( $this->debug & DBGSOCK ) echo "DBG.SOCK
chunked encoding..\n";
$length = fgets($this->socket, 1024);
$length = hexdec($length);
while (true) {
if ($length == 0) { break; }
$data .= fread($this->socket,
$length);
if( $this->debug & DBGSOCK )
echo "DBG.SOCK chunked encoding: read $length bytes\n";
fgets($this->socket, 1024);
$length = fgets($this->socket,
1024);
$length = hexdec($length);
}
fgets($this->socket, 1024);
}
else if ($this->responseHeaders['Content-Length'] )
{
$length =
$this->responseHeaders['Content-Length'];
$data = fread( $this->socket, $length );
if( $this->debug & DBGSOCK ) echo "DBG.SOCK
socket_read using Content-Length ($length)\n";
}
else {
if( $this->debug & DBGSOCK ) echo "Not chunked,
dont know how big?..\n";
$data = "";
$counter = 0;
socket_set_blocking( $this->socket, true );
socket_set_timeout($this->socket,2);
$ts1=time();
do{
$status = socket_get_status(
$this->socket );
/* if( $this->debug & DBGSOCK )
echo " Socket status:
"; print_r($status);
*/ if( feof($this->socket)) {
if( $this->debug & DBGSOCK )
echo "DBG.SOCK eof met, finished socket_read\n";
break;
}
if( $status['unread_bytes'] > 0 ) {
$buffer = fread( $this->socket,
$status['unread_bytes'] );
$counter = 0;
} else {
$ts=time();
$buffer = fread( $this->socket,
1024 );
sleep(0.1);
$failureCount++;
//print "elapsed
".(time()-$ts)."<br>";
}
$data .= $buffer;
} while( $status['unread_bytes'] > 0 ||
$counter++ < 10 );
//print "total ".(time()-$ts1)."<br>";
if( $this->debug & DBGSOCK ) {
echo "DBG.SOCK Counter:$counter\nRead
failure #: $failureCount\n";
echo " Socket status: ";
print_r($status);
}
socket_set_blocking( $this->socket, true );
}
$len = strlen($data);
if( $this->debug & DBGSOCK ) echo "DBG.SOCK read $len
bytes";
return $data;
}
/**
* Calculate and return the URI to be sent ( proxy purpose )
* @param the local URI
* @return URI to be used in the HTTP request
* @access private
*/
function makeUri( $uri )
{
$a = parse_url( $uri );
if( isset($a['scheme']) && isset($a['host']) ) {
$this->url = $a;
} else {
unset( $this->url['query']);
unset( $this->url['fragment']);
$this->url = array_merge( $this->url, $a );
}
if( $this->useProxy ) {
$requesturi= "http://" . $this->url['host'] . (
empty($this->url['port']) ? "" : ":" . $this->url['port'] ) .
$this->url['path'] . ( empty($this->url['query']) ? "" : "?" .
$this->url['query'] );
} else {
$requesturi = $this->url['path'] . (empty(
$this->url['query'] ) ? "" : "?" . $this->url['query']);
}
return $requesturi;
}
} // end class net_http_client
?>
- [Phpgroupware-cvs] phpgwapi/inc/class.net_http_client.inc.php, 1.3,
nomail <=