X-Git-Url: http://git.onelab.eu/?a=blobdiff_plain;f=tests%2F3LocalhostTest.php;h=4b721d6214e4084ab1126f3da0cfe198b87ce7a4;hb=a56b8ca18c8f34d2018d6f2099c25b3539804d19;hp=a77354864c4c3d0842b7bc2bbd0fa15742144ef8;hpb=e640c12e8f1cbc263f766035b0113ad132285e2a;p=plcapi.git diff --git a/tests/3LocalhostTest.php b/tests/3LocalhostTest.php index a773548..4b721d6 100644 --- a/tests/3LocalhostTest.php +++ b/tests/3LocalhostTest.php @@ -5,24 +5,32 @@ include_once __DIR__ . '/../lib/xmlrpc_wrappers.inc'; include_once __DIR__ . '/parse_args.php'; +/** + * Tests which involve interaction between the client and the server. + * They are run against the server found in demo/server.php + */ class LocalhostTest extends PHPUnit_Framework_TestCase { /** @var xmlrpc_client $client */ - public $client = null; - public $method = 'http'; - public $timeout = 10; - public $request_compression = null; - public $accepted_compression = ''; - public $args = array(); + protected $client = null; + protected $method = 'http'; + protected $timeout = 10; + protected $request_compression = null; + protected $accepted_compression = ''; + protected $args = array(); protected static $failed_tests = array(); + protected $testId; + /** @var boolean $collectCodeCoverageInformation */ + protected $collectCodeCoverageInformation; + protected $coverageScriptUrl; + public static function fail($message = '') { - // save in global var that this particular test has failed + // save in a static var that this particular test has failed // (but only if not called from subclass objects / multitests) if (function_exists('debug_backtrace') && strtolower(get_called_class()) == 'localhosttests') { - global $failed_tests; $trace = debug_backtrace(DEBUG_BACKTRACE_IGNORE_ARGS); for ($i = 0; $i < count($trace); $i++) { if (strpos($trace[$i]['function'], 'test') === 0) { @@ -36,8 +44,41 @@ class LocalhostTest extends PHPUnit_Framework_TestCase } /** - * @todo be smarter with setup, do not use global variables anymore + * Reimplemented to allow us to collect code coverage info from the target server. + * Code taken from PHPUnit_Extensions_Selenium2TestCase + * + * @param PHPUnit_Framework_TestResult $result + * @return PHPUnit_Framework_TestResult + * @throws Exception */ + public function run(PHPUnit_Framework_TestResult $result = NULL) + { + $this->testId = get_class($this) . '__' . $this->getName(); + + if ($result === NULL) { + $result = $this->createResult(); + } + + $this->collectCodeCoverageInformation = $result->getCollectCodeCoverageInformation(); + + parent::run($result); + + if ($this->collectCodeCoverageInformation) { + $coverage = new PHPUnit_Extensions_SeleniumCommon_RemoteCoverage( + $this->coverageScriptUrl, + $this->testId + ); + $result->getCodeCoverage()->append( + $coverage->get(), $this + ); + } + + // do not call this before to give the time to the Listeners to run + //$this->getStrategy()->endOfTest($this->session); + + return $result; + } + public function setUp() { $this->args = argParser::getArgs(); @@ -48,28 +89,47 @@ class LocalhostTest extends PHPUnit_Framework_TestCase } else { $this->client = new xmlrpc_client($this->args['URI'], $this->args['LOCALSERVER']); } - if ($this->args['DEBUG']) { - $this->client->setDebug($this->args['DEBUG']); - } + + $this->client->setDebug($this->args['DEBUG']); $this->client->request_compression = $this->request_compression; $this->client->accepted_compression = $this->accepted_compression; + + $this->coverageScriptUrl = 'http://' . $this->args['LOCALSERVER'] . '/' . str_replace( '/demo/server/server.php', 'tests/phpunit_coverage.php', $this->args['URI'] ); + + if ($this->args['DEBUG'] == 1) + ob_start(); } - public function send($msg, $errrorcode = 0, $return_response = false) + protected function tearDown() { + if ($this->args['DEBUG'] != 1) + return; + $out = ob_get_clean(); + $status = $this->getStatus(); + if ($status == PHPUnit_Runner_BaseTestRunner::STATUS_ERROR + || $status == PHPUnit_Runner_BaseTestRunner::STATUS_FAILURE) { + echo $out; + } + } + + protected function send($msg, $errorCode = 0, $returnResponse = false) + { + if ($this->collectCodeCoverageInformation) { + $this->client->setCookie('PHPUNIT_SELENIUM_TEST_ID', $this->testId); + } + $r = $this->client->send($msg, $this->timeout, $this->method); // for multicall, return directly array of responses if (is_array($r)) { return $r; } - if (is_array($errrorcode)) { - $this->assertContains($r->faultCode(), $errrorcode, 'Error ' . $r->faultCode() . ' connecting to server: ' . $r->faultString()); - } - else { - $this->assertEquals($r->faultCode(), $errrorcode, 'Error ' . $r->faultCode() . ' connecting to server: ' . $r->faultString()); + if (is_array($errorCode)) { + $this->assertContains($r->faultCode(), $errorCode, 'Error ' . $r->faultCode() . ' connecting to server: ' . $r->faultString()); + } else { + $this->assertEquals($errorCode, $r->faultCode(), 'Error ' . $r->faultCode() . ' connecting to server: ' . $r->faultString()); } if (!$r->faultCode()) { - if ($return_response) { + if ($returnResponse) { return $r; } else { return $r->value(); @@ -81,7 +141,7 @@ class LocalhostTest extends PHPUnit_Framework_TestCase public function testString() { - $sendstring = "here are 3 \"entities\": < > & " . + $sendString = "here are 3 \"entities\": < > & " . "and here's a dollar sign: \$pretendvarname and a backslash too: " . chr(92) . " - isn't that great? \\\"hackery\\\" at it's best " . " also don't want to miss out on \$item[0]. " . @@ -91,22 +151,59 @@ class LocalhostTest extends PHPUnit_Framework_TestCase "and then LFCR" . chr(10) . chr(13) . "last but not least weird names: G" . chr(252) . "nter, El" . chr(232) . "ne, and an xml comment closing tag: -->"; $f = new xmlrpcmsg('examples.stringecho', array( - new xmlrpcval($sendstring, 'string'), + new xmlrpcval($sendString, 'string'), )); $v = $this->send($f); if ($v) { // when sending/receiving non-US-ASCII encoded strings, XML says cr-lf can be normalized. // so we relax our tests... - $l1 = strlen($sendstring); + $l1 = strlen($sendString); $l2 = strlen($v->scalarval()); if ($l1 == $l2) { - $this->assertEquals($sendstring, $v->scalarval()); + $this->assertEquals($sendString, $v->scalarval()); } else { - $this->assertEquals(str_replace(array("\r\n", "\r"), array("\n", "\n"), $sendstring), $v->scalarval()); + $this->assertEquals(str_replace(array("\r\n", "\r"), array("\n", "\n"), $sendString), $v->scalarval()); } } } + public function testLatin1String() + { + $sendString = + "last but not least weird names: G" . chr(252) . "nter, El" . chr(232) . "ne"; + $f = 'examples.stringecho'. + $sendString. + ''; + $v = $this->send($f); + if ($v) { + $this->assertEquals($sendString, $v->scalarval()); + } + } + + /*public function testLatin1Method() + { + $f = new xmlrpcmsg("tests.iso88591methodname." . chr(224) . chr(252) . chr(232), array( + new xmlrpcval('hello') + )); + $v = $this->send($f); + if ($v) { + $this->assertEquals('hello', $v->scalarval()); + } + }*/ + + public function testUtf8Method() + { + PhpXmlRpc\PhpXmlRpc::$xmlrpc_internalencoding = 'UTF-8'; + $f = new xmlrpcmsg("tests.utf8methodname." . 'κόσμε', array( + new xmlrpcval('hello') + )); + $v = $this->send($f); + if ($v) { + $this->assertEquals('hello', $v->scalarval()); + } + PhpXmlRpc\PhpXmlRpc::$xmlrpc_internalencoding = 'ISO-8859-1'; + } + public function testAddingDoubles() { // note that rounding errors mean we @@ -156,9 +253,7 @@ class LocalhostTest extends PHPUnit_Framework_TestCase new xmlrpcval(true, 'boolean'), new xmlrpcval(false, 'boolean'), new xmlrpcval(1, 'boolean'), - new xmlrpcval(0, 'boolean'), - //new xmlrpcval('true', 'boolean'), - //new xmlrpcval('false', 'boolean') + new xmlrpcval(0, 'boolean') ), 'array' ),)); @@ -181,7 +276,7 @@ class LocalhostTest extends PHPUnit_Framework_TestCase public function testBase64() { - $sendstring = 'Mary had a little lamb, + $sendString = 'Mary had a little lamb, Whose fleece was white as snow, And everywhere that Mary went the lamb was sure to go. @@ -191,14 +286,14 @@ She tied it to a pylon Ten thousand volts went down its back And turned it into nylon'; $f = new xmlrpcmsg('examples.decode64', array( - new xmlrpcval($sendstring, 'base64'), + new xmlrpcval($sendString, 'base64'), )); $v = $this->send($f); if ($v) { - if (strlen($sendstring) == strlen($v->scalarval())) { - $this->assertEquals($sendstring, $v->scalarval()); + if (strlen($sendString) == strlen($v->scalarval())) { + $this->assertEquals($sendString, $v->scalarval()); } else { - $this->assertEquals(str_replace(array("\r\n", "\r"), array("\n", "\n"), $sendstring), $v->scalarval()); + $this->assertEquals(str_replace(array("\r\n", "\r"), array("\n", "\n"), $sendString), $v->scalarval()); } } } @@ -221,9 +316,9 @@ And turned it into nylon'; public function testCountEntities() { - $sendstring = "h'fd>onc>>l>>rw&bpu>q>esend($f); if ($v) { @@ -422,23 +517,23 @@ And turned it into nylon'; public function testCatchWarnings() { - $f = new xmlrpcmsg('examples.generatePHPWarning', array( + $f = new xmlrpcmsg('tests.generatePHPWarning', array( new xmlrpcval('whatever', 'string'), )); $v = $this->send($f); if ($v) { - $this->assertEquals($v->scalarval(), true); + $this->assertEquals(true, $v->scalarval()); } } public function testCatchExceptions() { - $f = new xmlrpcmsg('examples.raiseException', array( + $f = new xmlrpcmsg('tests.raiseException', array( new xmlrpcval('whatever', 'string'), )); $v = $this->send($f, $GLOBALS['xmlrpcerr']['server_error']); $this->client->path = $this->args['URI'] . '?EXCEPTION_HANDLING=1'; - $v = $this->send($f, 1); + $v = $this->send($f, 1); // the error code of the expected exception $this->client->path = $this->args['URI'] . '?EXCEPTION_HANDLING=2'; // depending on whether display_errors is ON or OFF on the server, we will get back a different error here, // as php will generate an http status code of either 200 or 500... @@ -456,61 +551,203 @@ And turned it into nylon'; $f = new xmlrpcmsg('system.MethodHelp'); $f->payload = "validator1.echoStructTest','')); echo('gotcha!'); die(); //"; $v = $this->send($f); - //$v = $r->faultCode(); if ($v) { $this->assertEquals(0, $v->structsize()); } } - public function testAutoRegisteredFunction() + public function testServerWrappedFunction() { - $f = new xmlrpcmsg('examples.php.getStateName', array( + $f = new xmlrpcmsg('tests.getStateName.2', array( new xmlrpcval(23, 'int'), )); $v = $this->send($f); - if ($v) { - $this->assertEquals('Michigan', $v->scalarval()); - } else { - $this->fail('Note: server can only auto register functions if running with PHP 5.0.3 and up'); - } + $this->assertEquals('Michigan', $v->scalarval()); + + // this generates an exception in the function which was wrapped, which is by default wrapped in a known error response + $f = new xmlrpcmsg('tests.getStateName.2', array( + new xmlrpcval(0, 'int'), + )); + $v = $this->send($f, $GLOBALS['xmlrpcerr']['server_error']); + + // check if the generated function dispatch map is fine, by checking if the server registered it + $f = new xmlrpcmsg('system.methodSignature', array( + new xmlrpcval('tests.getStateName.2'), + )); + $v = $this->send($f); + $encoder = new \PhpXmlRpc\Encoder(); + $this->assertEquals(array(array('string', 'int')), $encoder->decode($v)); } - public function testAutoRegisteredClass() + public function testServerWrappedFunctionAsSource() { - $f = new xmlrpcmsg('examples.php2.getStateName', array( + $f = new xmlrpcmsg('tests.getStateName.6', array( new xmlrpcval(23, 'int'), )); $v = $this->send($f); - if ($v) { - $this->assertEquals('Michigan', $v->scalarval()); - $f = new xmlrpcmsg('examples.php3.getStateName', array( - new xmlrpcval(23, 'int'), - )); - $v = $this->send($f); - if ($v) { - $this->assertEquals('Michigan', $v->scalarval()); - } + $this->assertEquals('Michigan', $v->scalarval()); + + // this generates an exception in the function which was wrapped, which is by default wrapped in a known error response + $f = new xmlrpcmsg('tests.getStateName.6', array( + new xmlrpcval(0, 'int'), + )); + $v = $this->send($f, $GLOBALS['xmlrpcerr']['server_error']); + } + + public function testServerWrappedObjectMethods() + { + $f = new xmlrpcmsg('tests.getStateName.3', array( + new xmlrpcval(23, 'int'), + )); + $v = $this->send($f); + $this->assertEquals('Michigan', $v->scalarval()); + + $f = new xmlrpcmsg('tests.getStateName.4', array( + new xmlrpcval(23, 'int'), + )); + $v = $this->send($f); + $this->assertEquals('Michigan', $v->scalarval()); + + $f = new xmlrpcmsg('tests.getStateName.5', array( + new xmlrpcval(23, 'int'), + )); + $v = $this->send($f); + $this->assertEquals('Michigan', $v->scalarval()); + + $f = new xmlrpcmsg('tests.getStateName.7', array( + new xmlrpcval(23, 'int'), + )); + $v = $this->send($f); + $this->assertEquals('Michigan', $v->scalarval()); + + $f = new xmlrpcmsg('tests.getStateName.8', array( + new xmlrpcval(23, 'int'), + )); + $v = $this->send($f); + $this->assertEquals('Michigan', $v->scalarval()); + + $f = new xmlrpcmsg('tests.getStateName.9', array( + new xmlrpcval(23, 'int'), + )); + $v = $this->send($f); + $this->assertEquals('Michigan', $v->scalarval()); + } + + public function testServerWrappedObjectMethodsAsSource() + { + $f = new xmlrpcmsg('tests.getStateName.7', array( + new xmlrpcval(23, 'int'), + )); + $v = $this->send($f); + $this->assertEquals('Michigan', $v->scalarval()); + + $f = new xmlrpcmsg('tests.getStateName.8', array( + new xmlrpcval(23, 'int'), + )); + $v = $this->send($f); + $this->assertEquals('Michigan', $v->scalarval()); + + $f = new xmlrpcmsg('tests.getStateName.9', array( + new xmlrpcval(23, 'int'), + )); + $v = $this->send($f); + $this->assertEquals('Michigan', $v->scalarval()); + } + + public function testServerClosure() + { + $f = new xmlrpcmsg('tests.getStateName.10', array( + new xmlrpcval(23, 'int'), + )); + $v = $this->send($f); + $this->assertEquals('Michigan', $v->scalarval()); + } + + public function testServerWrappedClosure() + { + $f = new xmlrpcmsg('tests.getStateName.11', array( + new xmlrpcval(23, 'int'), + )); + $v = $this->send($f); + $this->assertEquals('Michigan', $v->scalarval()); + } + + public function testServerWrappedClass() + { + $f = new xmlrpcmsg('tests.xmlrpcServerMethodsContainer.findState', array( + new xmlrpcval(23, 'int'), + )); + $v = $this->send($f); + $this->assertEquals('Michigan', $v->scalarval()); + } + + public function testWrappedMethod() + { + // make a 'deep client copy' as the original one might have many properties set + $func = wrap_xmlrpc_method($this->client, 'examples.getStateName', array('simple_client_copy' => 0)); + if ($func == false) { + $this->fail('Registration of examples.getStateName failed'); } else { - $this->fail('Note: server can only auto register class methods if running with PHP 5.0.3 and up'); + $v = $func(23); + // work around bug in current (or old?) version of phpunit when reporting the error + /*if (is_object($v)) { + $v = var_export($v, true); + }*/ + $this->assertEquals('Michigan', $v); } } - public function testAutoRegisteredMethod() + public function testWrappedMethodAsSource() { // make a 'deep client copy' as the original one might have many properties set - $func = wrap_xmlrpc_method($this->client, 'examples.getStateName', array('simple_client_copy' => 1)); - if ($func == '') { + $func = wrap_xmlrpc_method($this->client, 'examples.getStateName', array('simple_client_copy' => 0, 'return_source' => true)); + if ($func == false) { $this->fail('Registration of examples.getStateName failed'); } else { + eval($func['source']); + $func = $func['function']; $v = $func(23); - // work around bug in current version of phpunit - if (is_object($v)) { + // work around bug in current (or old?) version of phpunit when reporting the error + /*if (is_object($v)) { $v = var_export($v, true); - } + }*/ + $this->assertEquals('Michigan', $v); + } + } + + public function testWrappedClass() + { + // make a 'deep client copy' as the original one might have many properties set + // also for speed only wrap one method of the whole server + $class = wrap_xmlrpc_server($this->client, array('simple_client_copy' => 0, 'method_filter' => '/examples\.getStateName/' )); + if ($class == '') { + $this->fail('Registration of remote server failed'); + } else { + $obj = new $class(); + $v = $obj->examples_getStateName(23); + // work around bug in current (or old?) version of phpunit when reporting the error + /*if (is_object($v)) { + $v = var_export($v, true); + }*/ $this->assertEquals('Michigan', $v); } } + public function testTransferOfObjectViaWrapping() + { + // make a 'deep client copy' as the original one might have many properties set + $func = wrap_xmlrpc_method($this->client, 'tests.returnPhpObject', array('simple_client_copy' => true, + 'decode_php_objs' => true)); + if ($func == false) { + $this->fail('Registration of tests.returnPhpObject failed'); + } else { + $v = $func(); + $obj = new stdClass(); + $obj->hello = 'world'; + $this->assertEquals($obj, $v); + } + } + public function testGetCookies() { // let server set to us some cookies we tell it @@ -570,12 +807,18 @@ And turned it into nylon'; $cookies[$cookie] = (string)$cookies[$cookie]; } $r = $this->client->send($f, $this->timeout, $this->method); - $this->assertEquals($r->faultCode(), 0, 'Error ' . $r->faultCode() . ' connecting to server: ' . $r->faultString()); + $this->assertEquals(0, $r->faultCode(), 'Error ' . $r->faultCode() . ' connecting to server: ' . $r->faultString()); if (!$r->faultCode()) { $v = $r->value(); $v = php_xmlrpc_decode($v); + + // take care for the extra cookie used for coverage collection + if (isset($v['PHPUNIT_SELENIUM_TEST_ID'])) { + unset($v['PHPUNIT_SELENIUM_TEST_ID']); + } + // on IIS and Apache getallheaders returns something slightly different... - $this->assertEquals($v, $cookies); + $this->assertEquals($cookies, $v); } } @@ -586,9 +829,8 @@ And turned it into nylon'; )); $v1 = $this->send($f); $v2 = $this->send($f); - //$v = $r->faultCode(); if ($v1 && $v2) { - $this->assertEquals($v2, $v1); + $this->assertEquals($v1, $v2); } } }