1<?php 2/* 3 * 4 * Copyright 2015-2016 gRPC authors. 5 * 6 * Licensed under the Apache License, Version 2.0 (the "License"); 7 * you may not use this file except in compliance with the License. 8 * You may obtain a copy of the License at 9 * 10 * http://www.apache.org/licenses/LICENSE-2.0 11 * 12 * Unless required by applicable law or agreed to in writing, software 13 * distributed under the License is distributed on an "AS IS" BASIS, 14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 15 * See the License for the specific language governing permissions and 16 * limitations under the License. 17 * 18 */ 19require_once realpath(dirname(__FILE__).'/../../vendor/autoload.php'); 20 21// The following includes are needed when using protobuf 3.1.0 22// and will suppress warnings when using protobuf 3.2.0+ 23@include_once 'src/proto/grpc/testing/test.pb.php'; 24@include_once 'src/proto/grpc/testing/test_grpc_pb.php'; 25 26use Google\Auth\CredentialsLoader; 27use Google\Auth\ApplicationDefaultCredentials; 28use GuzzleHttp\ClientInterface; 29 30/** 31 * Assertion function that always exits with an error code if the assertion is 32 * falsy. 33 * 34 * @param $value Assertion value. Should be true. 35 * @param $error_message Message to display if the assertion is false 36 */ 37function hardAssert($value, $error_message) 38{ 39 if (!$value) { 40 echo $error_message."\n"; 41 exit(1); 42 } 43} 44 45function hardAssertIfStatusOk($status) 46{ 47 if ($status->code !== Grpc\STATUS_OK) { 48 echo "Call did not complete successfully. Status object:\n"; 49 var_dump($status); 50 exit(1); 51 } 52} 53 54/** 55 * Run the empty_unary test. 56 * 57 * @param $stub Stub object that has service methods 58 */ 59function emptyUnary($stub) 60{ 61 list($result, $status) = 62 $stub->EmptyCall(new Grpc\Testing\EmptyMessage())->wait(); 63 hardAssertIfStatusOk($status); 64 hardAssert($result !== null, 'Call completed with a null response'); 65} 66 67/** 68 * Run the large_unary test. 69 * 70 * @param $stub Stub object that has service methods 71 */ 72function largeUnary($stub) 73{ 74 performLargeUnary($stub); 75} 76 77/** 78 * Shared code between large unary test and auth test. 79 * 80 * @param $stub Stub object that has service methods 81 * @param $fillUsername boolean whether to fill result with username 82 * @param $fillOauthScope boolean whether to fill result with oauth scope 83 */ 84function performLargeUnary($stub, $fillUsername = false, 85 $fillOauthScope = false, $callback = false) 86{ 87 $request_len = 271828; 88 $response_len = 314159; 89 90 $request = new Grpc\Testing\SimpleRequest(); 91 $request->setResponseType(Grpc\Testing\PayloadType::COMPRESSABLE); 92 $request->setResponseSize($response_len); 93 $payload = new Grpc\Testing\Payload(); 94 $payload->setType(Grpc\Testing\PayloadType::COMPRESSABLE); 95 $payload->setBody(str_repeat("\0", $request_len)); 96 $request->setPayload($payload); 97 $request->setFillUsername($fillUsername); 98 $request->setFillOauthScope($fillOauthScope); 99 100 $options = []; 101 if ($callback) { 102 $options['call_credentials_callback'] = $callback; 103 } 104 105 list($result, $status) = $stub->UnaryCall($request, [], $options)->wait(); 106 hardAssertIfStatusOk($status); 107 hardAssert($result !== null, 'Call returned a null response'); 108 $payload = $result->getPayload(); 109 hardAssert($payload->getType() === Grpc\Testing\PayloadType::COMPRESSABLE, 110 'Payload had the wrong type'); 111 hardAssert(strlen($payload->getBody()) === $response_len, 112 'Payload had the wrong length'); 113 hardAssert($payload->getBody() === str_repeat("\0", $response_len), 114 'Payload had the wrong content'); 115 116 return $result; 117} 118 119/** 120 * Run the service account credentials auth test. 121 * 122 * @param $stub Stub object that has service methods 123 * @param $args array command line args 124 */ 125function serviceAccountCreds($stub, $args) 126{ 127 if (!array_key_exists('oauth_scope', $args)) { 128 throw new Exception('Missing oauth scope'); 129 } 130 $jsonKey = json_decode( 131 file_get_contents(getenv(CredentialsLoader::ENV_VAR)), 132 true); 133 $result = performLargeUnary($stub, $fillUsername = true, 134 $fillOauthScope = true); 135 hardAssert($result->getUsername() === $jsonKey['client_email'], 136 'invalid email returned'); 137 hardAssert(strpos($args['oauth_scope'], $result->getOauthScope()) !== false, 138 'invalid oauth scope returned'); 139} 140 141/** 142 * Run the compute engine credentials auth test. 143 * Has not been run from gcloud as of 2015-05-05. 144 * 145 * @param $stub Stub object that has service methods 146 * @param $args array command line args 147 */ 148function computeEngineCreds($stub, $args) 149{ 150 if (!array_key_exists('oauth_scope', $args)) { 151 throw new Exception('Missing oauth scope'); 152 } 153 if (!array_key_exists('default_service_account', $args)) { 154 throw new Exception('Missing default_service_account'); 155 } 156 $result = performLargeUnary($stub, $fillUsername = true, 157 $fillOauthScope = true); 158 hardAssert($args['default_service_account'] === $result->getUsername(), 159 'invalid email returned'); 160} 161 162/** 163 * Run the jwt token credentials auth test. 164 * 165 * @param $stub Stub object that has service methods 166 * @param $args array command line args 167 */ 168function jwtTokenCreds($stub, $args) 169{ 170 $jsonKey = json_decode( 171 file_get_contents(getenv(CredentialsLoader::ENV_VAR)), 172 true); 173 $result = performLargeUnary($stub, $fillUsername = true, 174 $fillOauthScope = true); 175 hardAssert($result->getUsername() === $jsonKey['client_email'], 176 'invalid email returned'); 177} 178 179/** 180 * Run the oauth2_auth_token auth test. 181 * 182 * @param $stub Stub object that has service methods 183 * @param $args array command line args 184 */ 185function oauth2AuthToken($stub, $args) 186{ 187 $jsonKey = json_decode( 188 file_get_contents(getenv(CredentialsLoader::ENV_VAR)), 189 true); 190 $result = performLargeUnary($stub, $fillUsername = true, 191 $fillOauthScope = true); 192 hardAssert($result->getUsername() === $jsonKey['client_email'], 193 'invalid email returned'); 194} 195 196function updateAuthMetadataCallback($context) 197{ 198 $authUri = $context->service_url; 199 $methodName = $context->method_name; 200 $auth_credentials = ApplicationDefaultCredentials::getCredentials(); 201 202 $metadata = []; 203 $result = $auth_credentials->updateMetadata([], $authUri); 204 foreach ($result as $key => $value) { 205 $metadata[strtolower($key)] = $value; 206 } 207 208 return $metadata; 209} 210 211/** 212 * Run the per_rpc_creds auth test. 213 * 214 * @param $stub Stub object that has service methods 215 * @param $args array command line args 216 */ 217function perRpcCreds($stub, $args) 218{ 219 $jsonKey = json_decode( 220 file_get_contents(getenv(CredentialsLoader::ENV_VAR)), 221 true); 222 223 $result = performLargeUnary($stub, $fillUsername = true, 224 $fillOauthScope = true, 225 'updateAuthMetadataCallback'); 226 hardAssert($result->getUsername() === $jsonKey['client_email'], 227 'invalid email returned'); 228} 229 230/** 231 * Run the client_streaming test. 232 * 233 * @param $stub Stub object that has service methods 234 */ 235function clientStreaming($stub) 236{ 237 $request_lengths = [27182, 8, 1828, 45904]; 238 239 $requests = array_map( 240 function ($length) { 241 $request = new Grpc\Testing\StreamingInputCallRequest(); 242 $payload = new Grpc\Testing\Payload(); 243 $payload->setBody(str_repeat("\0", $length)); 244 $request->setPayload($payload); 245 246 return $request; 247 }, $request_lengths); 248 249 $call = $stub->StreamingInputCall(); 250 foreach ($requests as $request) { 251 $call->write($request); 252 } 253 list($result, $status) = $call->wait(); 254 hardAssertIfStatusOk($status); 255 hardAssert($result->getAggregatedPayloadSize() === 74922, 256 'aggregated_payload_size was incorrect'); 257} 258 259/** 260 * Run the server_streaming test. 261 * 262 * @param $stub Stub object that has service methods. 263 */ 264function serverStreaming($stub) 265{ 266 $sizes = [31415, 9, 2653, 58979]; 267 268 $request = new Grpc\Testing\StreamingOutputCallRequest(); 269 $request->setResponseType(Grpc\Testing\PayloadType::COMPRESSABLE); 270 foreach ($sizes as $size) { 271 $response_parameters = new Grpc\Testing\ResponseParameters(); 272 $response_parameters->setSize($size); 273 $request->getResponseParameters()[] = $response_parameters; 274 } 275 276 $call = $stub->StreamingOutputCall($request); 277 $i = 0; 278 foreach ($call->responses() as $value) { 279 hardAssert($i < 4, 'Too many responses'); 280 $payload = $value->getPayload(); 281 hardAssert( 282 $payload->getType() === Grpc\Testing\PayloadType::COMPRESSABLE, 283 'Payload '.$i.' had the wrong type'); 284 hardAssert(strlen($payload->getBody()) === $sizes[$i], 285 'Response '.$i.' had the wrong length'); 286 $i += 1; 287 } 288 hardAssertIfStatusOk($call->getStatus()); 289} 290 291/** 292 * Run the ping_pong test. 293 * 294 * @param $stub Stub object that has service methods. 295 */ 296function pingPong($stub) 297{ 298 $request_lengths = [27182, 8, 1828, 45904]; 299 $response_lengths = [31415, 9, 2653, 58979]; 300 301 $call = $stub->FullDuplexCall(); 302 for ($i = 0; $i < 4; ++$i) { 303 $request = new Grpc\Testing\StreamingOutputCallRequest(); 304 $request->setResponseType(Grpc\Testing\PayloadType::COMPRESSABLE); 305 $response_parameters = new Grpc\Testing\ResponseParameters(); 306 $response_parameters->setSize($response_lengths[$i]); 307 $request->getResponseParameters()[] = $response_parameters; 308 $payload = new Grpc\Testing\Payload(); 309 $payload->setBody(str_repeat("\0", $request_lengths[$i])); 310 $request->setPayload($payload); 311 312 $call->write($request); 313 $response = $call->read(); 314 315 hardAssert($response !== null, 'Server returned too few responses'); 316 $payload = $response->getPayload(); 317 hardAssert( 318 $payload->getType() === Grpc\Testing\PayloadType::COMPRESSABLE, 319 'Payload '.$i.' had the wrong type'); 320 hardAssert(strlen($payload->getBody()) === $response_lengths[$i], 321 'Payload '.$i.' had the wrong length'); 322 } 323 $call->writesDone(); 324 hardAssert($call->read() === null, 'Server returned too many responses'); 325 hardAssertIfStatusOk($call->getStatus()); 326} 327 328/** 329 * Run the empty_stream test. 330 * 331 * @param $stub Stub object that has service methods. 332 */ 333function emptyStream($stub) 334{ 335 $call = $stub->FullDuplexCall(); 336 $call->writesDone(); 337 hardAssert($call->read() === null, 'Server returned too many responses'); 338 hardAssertIfStatusOk($call->getStatus()); 339} 340 341/** 342 * Run the cancel_after_begin test. 343 * 344 * @param $stub Stub object that has service methods. 345 */ 346function cancelAfterBegin($stub) 347{ 348 $call = $stub->StreamingInputCall(); 349 $call->cancel(); 350 list($result, $status) = $call->wait(); 351 hardAssert($status->code === Grpc\STATUS_CANCELLED, 352 'Call status was not CANCELLED'); 353} 354 355/** 356 * Run the cancel_after_first_response test. 357 * 358 * @param $stub Stub object that has service methods. 359 */ 360function cancelAfterFirstResponse($stub) 361{ 362 $call = $stub->FullDuplexCall(); 363 $request = new Grpc\Testing\StreamingOutputCallRequest(); 364 $request->setResponseType(Grpc\Testing\PayloadType::COMPRESSABLE); 365 $response_parameters = new Grpc\Testing\ResponseParameters(); 366 $response_parameters->setSize(31415); 367 $request->getResponseParameters()[] = $response_parameters; 368 $payload = new Grpc\Testing\Payload(); 369 $payload->setBody(str_repeat("\0", 27182)); 370 $request->setPayload($payload); 371 372 $call->write($request); 373 $response = $call->read(); 374 375 $call->cancel(); 376 hardAssert($call->getStatus()->code === Grpc\STATUS_CANCELLED, 377 'Call status was not CANCELLED'); 378} 379 380function timeoutOnSleepingServer($stub) 381{ 382 $call = $stub->FullDuplexCall([], ['timeout' => 1000]); 383 $request = new Grpc\Testing\StreamingOutputCallRequest(); 384 $request->setResponseType(Grpc\Testing\PayloadType::COMPRESSABLE); 385 $response_parameters = new Grpc\Testing\ResponseParameters(); 386 $response_parameters->setSize(8); 387 $request->getResponseParameters()[] = $response_parameters; 388 $payload = new Grpc\Testing\Payload(); 389 $payload->setBody(str_repeat("\0", 9)); 390 $request->setPayload($payload); 391 392 $call->write($request); 393 $response = $call->read(); 394 395 hardAssert($call->getStatus()->code === Grpc\STATUS_DEADLINE_EXCEEDED, 396 'Call status was not DEADLINE_EXCEEDED'); 397} 398 399function customMetadata($stub) 400{ 401 $ECHO_INITIAL_KEY = 'x-grpc-test-echo-initial'; 402 $ECHO_INITIAL_VALUE = 'test_initial_metadata_value'; 403 $ECHO_TRAILING_KEY = 'x-grpc-test-echo-trailing-bin'; 404 $ECHO_TRAILING_VALUE = 'ababab'; 405 $request_len = 271828; 406 $response_len = 314159; 407 408 $request = new Grpc\Testing\SimpleRequest(); 409 $request->setResponseType(Grpc\Testing\PayloadType::COMPRESSABLE); 410 $request->setResponseSize($response_len); 411 $payload = new Grpc\Testing\Payload(); 412 $payload->setType(Grpc\Testing\PayloadType::COMPRESSABLE); 413 $payload->setBody(str_repeat("\0", $request_len)); 414 $request->setPayload($payload); 415 416 $metadata = [ 417 $ECHO_INITIAL_KEY => [$ECHO_INITIAL_VALUE], 418 $ECHO_TRAILING_KEY => [$ECHO_TRAILING_VALUE], 419 ]; 420 $call = $stub->UnaryCall($request, $metadata); 421 422 $initial_metadata = $call->getMetadata(); 423 hardAssert(array_key_exists($ECHO_INITIAL_KEY, $initial_metadata), 424 'Initial metadata does not contain expected key'); 425 hardAssert( 426 $initial_metadata[$ECHO_INITIAL_KEY][0] === $ECHO_INITIAL_VALUE, 427 'Incorrect initial metadata value'); 428 429 list($result, $status) = $call->wait(); 430 hardAssertIfStatusOk($status); 431 432 $trailing_metadata = $call->getTrailingMetadata(); 433 hardAssert(array_key_exists($ECHO_TRAILING_KEY, $trailing_metadata), 434 'Trailing metadata does not contain expected key'); 435 hardAssert( 436 $trailing_metadata[$ECHO_TRAILING_KEY][0] === $ECHO_TRAILING_VALUE, 437 'Incorrect trailing metadata value'); 438 439 $streaming_call = $stub->FullDuplexCall($metadata); 440 441 $streaming_request = new Grpc\Testing\StreamingOutputCallRequest(); 442 $streaming_request->setPayload($payload); 443 $response_parameters = new Grpc\Testing\ResponseParameters(); 444 $response_parameters->setSize($response_len); 445 $streaming_request->getResponseParameters()[] = $response_parameters; 446 $streaming_call->write($streaming_request); 447 $streaming_call->writesDone(); 448 $result = $streaming_call->read(); 449 450 hardAssertIfStatusOk($streaming_call->getStatus()); 451 452 $streaming_initial_metadata = $streaming_call->getMetadata(); 453 hardAssert(array_key_exists($ECHO_INITIAL_KEY, $streaming_initial_metadata), 454 'Initial metadata does not contain expected key'); 455 hardAssert( 456 $streaming_initial_metadata[$ECHO_INITIAL_KEY][0] === $ECHO_INITIAL_VALUE, 457 'Incorrect initial metadata value'); 458 459 $streaming_trailing_metadata = $streaming_call->getTrailingMetadata(); 460 hardAssert(array_key_exists($ECHO_TRAILING_KEY, 461 $streaming_trailing_metadata), 462 'Trailing metadata does not contain expected key'); 463 hardAssert($streaming_trailing_metadata[$ECHO_TRAILING_KEY][0] === 464 $ECHO_TRAILING_VALUE, 'Incorrect trailing metadata value'); 465} 466 467function statusCodeAndMessage($stub) 468{ 469 $echo_status = new Grpc\Testing\EchoStatus(); 470 $echo_status->setCode(2); 471 $echo_status->setMessage('test status message'); 472 473 $request = new Grpc\Testing\SimpleRequest(); 474 $request->setResponseStatus($echo_status); 475 476 $call = $stub->UnaryCall($request); 477 list($result, $status) = $call->wait(); 478 479 hardAssert($status->code === 2, 480 'Received unexpected UnaryCall status code: '. 481 $status->code); 482 hardAssert($status->details === 'test status message', 483 'Received unexpected UnaryCall status details: '. 484 $status->details); 485 486 $streaming_call = $stub->FullDuplexCall(); 487 488 $streaming_request = new Grpc\Testing\StreamingOutputCallRequest(); 489 $streaming_request->setResponseStatus($echo_status); 490 $streaming_call->write($streaming_request); 491 $streaming_call->writesDone(); 492 $result = $streaming_call->read(); 493 494 $status = $streaming_call->getStatus(); 495 hardAssert($status->code === 2, 496 'Received unexpected FullDuplexCall status code: '. 497 $status->code); 498 hardAssert($status->details === 'test status message', 499 'Received unexpected FullDuplexCall status details: '. 500 $status->details); 501} 502 503# NOTE: the stub input to this function is from UnimplementedService 504function unimplementedService($stub) 505{ 506 $call = $stub->UnimplementedCall(new Grpc\Testing\EmptyMessage()); 507 list($result, $status) = $call->wait(); 508 hardAssert($status->code === Grpc\STATUS_UNIMPLEMENTED, 509 'Received unexpected status code'); 510} 511 512# NOTE: the stub input to this function is from TestService 513function unimplementedMethod($stub) 514{ 515 $call = $stub->UnimplementedCall(new Grpc\Testing\EmptyMessage()); 516 list($result, $status) = $call->wait(); 517 hardAssert($status->code === Grpc\STATUS_UNIMPLEMENTED, 518 'Received unexpected status code'); 519} 520 521function _makeStub($args) 522{ 523 if (!array_key_exists('server_host', $args)) { 524 throw new Exception('Missing argument: --server_host is required'); 525 } 526 if (!array_key_exists('server_port', $args)) { 527 throw new Exception('Missing argument: --server_port is required'); 528 } 529 if (!array_key_exists('test_case', $args)) { 530 throw new Exception('Missing argument: --test_case is required'); 531 } 532 533 if ($args['server_port'] === 443) { 534 $server_address = $args['server_host']; 535 } else { 536 $server_address = $args['server_host'].':'.$args['server_port']; 537 } 538 539 $test_case = $args['test_case']; 540 541 $host_override = 'foo.test.google.fr'; 542 if (array_key_exists('server_host_override', $args)) { 543 $host_override = $args['server_host_override']; 544 } 545 546 $use_tls = false; 547 if (array_key_exists('use_tls', $args) && 548 $args['use_tls'] != 'false') { 549 $use_tls = true; 550 } 551 552 $use_test_ca = false; 553 if (array_key_exists('use_test_ca', $args) && 554 $args['use_test_ca'] != 'false') { 555 $use_test_ca = true; 556 } 557 558 $opts = []; 559 560 if ($use_tls) { 561 if ($use_test_ca) { 562 $ssl_credentials = Grpc\ChannelCredentials::createSsl( 563 file_get_contents(dirname(__FILE__).'/../data/ca.pem')); 564 } else { 565 $ssl_credentials = Grpc\ChannelCredentials::createSsl(); 566 } 567 $opts['credentials'] = $ssl_credentials; 568 $opts['grpc.ssl_target_name_override'] = $host_override; 569 } else { 570 $opts['credentials'] = Grpc\ChannelCredentials::createInsecure(); 571 } 572 573 if (in_array($test_case, ['service_account_creds', 574 'compute_engine_creds', 'jwt_token_creds', ])) { 575 if ($test_case === 'jwt_token_creds') { 576 $auth_credentials = ApplicationDefaultCredentials::getCredentials(); 577 } else { 578 $auth_credentials = ApplicationDefaultCredentials::getCredentials( 579 $args['oauth_scope'] 580 ); 581 } 582 $opts['update_metadata'] = $auth_credentials->getUpdateMetadataFunc(); 583 } 584 585 if ($test_case === 'oauth2_auth_token') { 586 $auth_credentials = ApplicationDefaultCredentials::getCredentials( 587 $args['oauth_scope'] 588 ); 589 $token = $auth_credentials->fetchAuthToken(); 590 $update_metadata = 591 function ($metadata, 592 $authUri = null, 593 ClientInterface $client = null) use ($token) { 594 $metadata_copy = $metadata; 595 $metadata_copy[CredentialsLoader::AUTH_METADATA_KEY] = 596 [sprintf('%s %s', 597 $token['token_type'], 598 $token['access_token'])]; 599 600 return $metadata_copy; 601 }; 602 $opts['update_metadata'] = $update_metadata; 603 } 604 605 if ($test_case === 'unimplemented_service') { 606 $stub = new Grpc\Testing\UnimplementedServiceClient($server_address, 607 $opts); 608 } else { 609 $stub = new Grpc\Testing\TestServiceClient($server_address, $opts); 610 } 611 612 return $stub; 613} 614 615function interop_main($args, $stub = false) 616{ 617 if (!$stub) { 618 $stub = _makeStub($args); 619 } 620 621 $test_case = $args['test_case']; 622 echo "Running test case $test_case\n"; 623 624 switch ($test_case) { 625 case 'empty_unary': 626 emptyUnary($stub); 627 break; 628 case 'large_unary': 629 largeUnary($stub); 630 break; 631 case 'client_streaming': 632 clientStreaming($stub); 633 break; 634 case 'server_streaming': 635 serverStreaming($stub); 636 break; 637 case 'ping_pong': 638 pingPong($stub); 639 break; 640 case 'empty_stream': 641 emptyStream($stub); 642 break; 643 case 'cancel_after_begin': 644 cancelAfterBegin($stub); 645 break; 646 case 'cancel_after_first_response': 647 cancelAfterFirstResponse($stub); 648 break; 649 case 'timeout_on_sleeping_server': 650 timeoutOnSleepingServer($stub); 651 break; 652 case 'custom_metadata': 653 customMetadata($stub); 654 break; 655 case 'status_code_and_message': 656 statusCodeAndMessage($stub); 657 break; 658 case 'unimplemented_service': 659 unimplementedService($stub); 660 break; 661 case 'unimplemented_method': 662 unimplementedMethod($stub); 663 break; 664 case 'service_account_creds': 665 serviceAccountCreds($stub, $args); 666 break; 667 case 'compute_engine_creds': 668 computeEngineCreds($stub, $args); 669 break; 670 case 'jwt_token_creds': 671 jwtTokenCreds($stub, $args); 672 break; 673 case 'oauth2_auth_token': 674 oauth2AuthToken($stub, $args); 675 break; 676 case 'per_rpc_creds': 677 perRpcCreds($stub, $args); 678 break; 679 default: 680 echo "Unsupported test case $test_case\n"; 681 exit(1); 682 } 683 684 return $stub; 685} 686 687if (isset($_SERVER['PHP_SELF']) && 688 preg_match('/interop_client/', $_SERVER['PHP_SELF'])) { 689 $args = getopt('', ['server_host:', 'server_port:', 'test_case:', 690 'use_tls::', 'use_test_ca::', 691 'server_host_override:', 'oauth_scope:', 692 'default_service_account:', ]); 693 interop_main($args); 694} 695