hostname = $hostname; $this->update_metadata = null; if (isset($opts['update_metadata'])) { if (is_callable($opts['update_metadata'])) { $this->update_metadata = $opts['update_metadata']; } unset($opts['update_metadata']); } if (!empty($opts['grpc.ssl_target_name_override'])) { $this->hostname_override = $opts['grpc.ssl_target_name_override']; } if ($channel) { if (!is_a($channel, 'Grpc\Channel') && !is_a($channel, 'Grpc\Internal\InterceptorChannel')) { throw new \Exception('The channel argument is not a Channel object '. 'or an InterceptorChannel object created by '. 'Interceptor::intercept($channel, Interceptor|Interceptor[] $interceptors)'); } $this->channel = $channel; return; } $this->channel = static::getDefaultChannel($hostname, $opts); } /** * Creates and returns the default Channel * * @param array $opts Channel constructor options * * @return Channel The channel */ public static function getDefaultChannel($hostname, array $opts) { $package_config = json_decode( file_get_contents(dirname(__FILE__).'/../../composer.json'), true ); if (!empty($opts['grpc.primary_user_agent'])) { $opts['grpc.primary_user_agent'] .= ' '; } else { $opts['grpc.primary_user_agent'] = ''; } $opts['grpc.primary_user_agent'] .= 'grpc-php/'.$package_config['version']; if (!array_key_exists('credentials', $opts)) { throw new \Exception("The opts['credentials'] key is now ". 'required. Please see one of the '. 'ChannelCredentials::create methods'); } return new Channel($hostname, $opts); } /** * @return string The URI of the endpoint */ public function getTarget() { return $this->channel->getTarget(); } /** * @param bool $try_to_connect (optional) * * @return int The grpc connectivity state */ public function getConnectivityState($try_to_connect = false) { return $this->channel->getConnectivityState($try_to_connect); } /** * @param int $timeout in microseconds * * @return bool true if channel is ready * @throw Exception if channel is in FATAL_ERROR state */ public function waitForReady($timeout) { $new_state = $this->getConnectivityState(true); if ($this->_checkConnectivityState($new_state)) { return true; } $now = Timeval::now(); $delta = new Timeval($timeout); $deadline = $now->add($delta); while ($this->channel->watchConnectivityState($new_state, $deadline)) { // state has changed before deadline $new_state = $this->getConnectivityState(); if ($this->_checkConnectivityState($new_state)) { return true; } } // deadline has passed $new_state = $this->getConnectivityState(); return $this->_checkConnectivityState($new_state); } /** * Close the communication channel associated with this stub. */ public function close() { $this->channel->close(); } /** * @param $new_state Connect state * * @return bool true if state is CHANNEL_READY * @throw Exception if state is CHANNEL_FATAL_FAILURE */ private function _checkConnectivityState($new_state) { if ($new_state == \Grpc\CHANNEL_READY) { return true; } if ($new_state == \Grpc\CHANNEL_FATAL_FAILURE) { throw new \Exception('Failed to connect to server'); } return false; } /** * constructs the auth uri for the jwt. * * @param string $method The method string * * @return string The URL string */ private function _get_jwt_aud_uri($method) { $last_slash_idx = strrpos($method, '/'); if ($last_slash_idx === false) { throw new \InvalidArgumentException( 'service name must have a slash' ); } $service_name = substr($method, 0, $last_slash_idx); if ($this->hostname_override) { $hostname = $this->hostname_override; } else { $hostname = $this->hostname; } return 'https://'.$hostname.$service_name; } /** * validate and normalize the metadata array. * * @param array $metadata The metadata map * * @return array $metadata Validated and key-normalized metadata map * @throw InvalidArgumentException if key contains invalid characters */ private function _validate_and_normalize_metadata($metadata) { $metadata_copy = []; foreach ($metadata as $key => $value) { if (!preg_match('/^[A-Za-z\d_-]+$/', $key)) { throw new \InvalidArgumentException( 'Metadata keys must be nonempty strings containing only '. 'alphanumeric characters, hyphens and underscores' ); } $metadata_copy[strtolower($key)] = $value; } return $metadata_copy; } /** * Create a function which can be used to create UnaryCall * * @param Channel|InterceptorChannel $channel * @param callable $deserialize A function that deserializes the response * * @return \Closure */ private function _GrpcUnaryUnary($channel) { return function ($method, $argument, $deserialize, array $metadata = [], array $options = []) use ($channel) { $call = new UnaryCall( $channel, $method, $deserialize, $options ); $jwt_aud_uri = $this->_get_jwt_aud_uri($method); if (is_callable($this->update_metadata)) { $metadata = call_user_func( $this->update_metadata, $metadata, $jwt_aud_uri ); } $metadata = $this->_validate_and_normalize_metadata( $metadata ); $call->start($argument, $metadata, $options); return $call; }; } /** * Create a function which can be used to create ServerStreamingCall * * @param Channel|InterceptorChannel $channel * @param callable $deserialize A function that deserializes the response * * @return \Closure */ private function _GrpcStreamUnary($channel) { return function ($method, $deserialize, array $metadata = [], array $options = []) use ($channel) { $call = new ClientStreamingCall( $channel, $method, $deserialize, $options ); $jwt_aud_uri = $this->_get_jwt_aud_uri($method); if (is_callable($this->update_metadata)) { $metadata = call_user_func( $this->update_metadata, $metadata, $jwt_aud_uri ); } $metadata = $this->_validate_and_normalize_metadata( $metadata ); $call->start($metadata); return $call; }; } /** * Create a function which can be used to create ClientStreamingCall * * @param Channel|InterceptorChannel $channel * @param callable $deserialize A function that deserializes the response * * @return \Closure */ private function _GrpcUnaryStream($channel) { return function ($method, $argument, $deserialize, array $metadata = [], array $options = []) use ($channel) { $call = new ServerStreamingCall( $channel, $method, $deserialize, $options ); $jwt_aud_uri = $this->_get_jwt_aud_uri($method); if (is_callable($this->update_metadata)) { $metadata = call_user_func( $this->update_metadata, $metadata, $jwt_aud_uri ); } $metadata = $this->_validate_and_normalize_metadata( $metadata ); $call->start($argument, $metadata, $options); return $call; }; } /** * Create a function which can be used to create BidiStreamingCall * * @param Channel|InterceptorChannel $channel * @param callable $deserialize A function that deserializes the response * * @return \Closure */ private function _GrpcStreamStream($channel) { return function ($method, $deserialize, array $metadata = [], array $options = []) use ($channel) { $call = new BidiStreamingCall( $channel, $method, $deserialize, $options ); $jwt_aud_uri = $this->_get_jwt_aud_uri($method); if (is_callable($this->update_metadata)) { $metadata = call_user_func( $this->update_metadata, $metadata, $jwt_aud_uri ); } $metadata = $this->_validate_and_normalize_metadata( $metadata ); $call->start($metadata); return $call; }; } /** * Create a function which can be used to create UnaryCall * * @param Channel|InterceptorChannel $channel * @param callable $deserialize A function that deserializes the response * * @return \Closure */ private function _UnaryUnaryCallFactory($channel) { if (is_a($channel, 'Grpc\Internal\InterceptorChannel')) { return function ($method, $argument, $deserialize, array $metadata = [], array $options = []) use ($channel) { return $channel->getInterceptor()->interceptUnaryUnary( $method, $argument, $deserialize, $metadata, $options, $this->_UnaryUnaryCallFactory($channel->getNext()) ); }; } return $this->_GrpcUnaryUnary($channel); } /** * Create a function which can be used to create ServerStreamingCall * * @param Channel|InterceptorChannel $channel * @param callable $deserialize A function that deserializes the response * * @return \Closure */ private function _UnaryStreamCallFactory($channel) { if (is_a($channel, 'Grpc\Internal\InterceptorChannel')) { return function ($method, $argument, $deserialize, array $metadata = [], array $options = []) use ($channel) { return $channel->getInterceptor()->interceptUnaryStream( $method, $argument, $deserialize, $metadata, $options, $this->_UnaryStreamCallFactory($channel->getNext()) ); }; } return $this->_GrpcUnaryStream($channel); } /** * Create a function which can be used to create ClientStreamingCall * * @param Channel|InterceptorChannel $channel * @param callable $deserialize A function that deserializes the response * * @return \Closure */ private function _StreamUnaryCallFactory($channel) { if (is_a($channel, 'Grpc\Internal\InterceptorChannel')) { return function ($method, $deserialize, array $metadata = [], array $options = []) use ($channel) { return $channel->getInterceptor()->interceptStreamUnary( $method, $deserialize, $metadata, $options, $this->_StreamUnaryCallFactory($channel->getNext()) ); }; } return $this->_GrpcStreamUnary($channel); } /** * Create a function which can be used to create BidiStreamingCall * * @param Channel|InterceptorChannel $channel * @param callable $deserialize A function that deserializes the response * * @return \Closure */ private function _StreamStreamCallFactory($channel) { if (is_a($channel, 'Grpc\Internal\InterceptorChannel')) { return function ($method, $deserialize, array $metadata = [], array $options = []) use ($channel) { return $channel->getInterceptor()->interceptStreamStream( $method, $deserialize, $metadata, $options, $this->_StreamStreamCallFactory($channel->getNext()) ); }; } return $this->_GrpcStreamStream($channel); } /* This class is intended to be subclassed by generated code, so * all functions begin with "_" to avoid name collisions. */ /** * Call a remote method that takes a single argument and has a * single output. * * @param string $method The name of the method to call * @param mixed $argument The argument to the method * @param callable $deserialize A function that deserializes the response * @param array $metadata A metadata map to send to the server * (optional) * @param array $options An array of options (optional) * * @return UnaryCall The active call object */ protected function _simpleRequest( $method, $argument, $deserialize, array $metadata = [], array $options = [] ) { $call_factory = $this->_UnaryUnaryCallFactory($this->channel); $call = $call_factory($method, $argument, $deserialize, $metadata, $options); return $call; } /** * Call a remote method that takes a stream of arguments and has a single * output. * * @param string $method The name of the method to call * @param callable $deserialize A function that deserializes the response * @param array $metadata A metadata map to send to the server * (optional) * @param array $options An array of options (optional) * * @return ClientStreamingCall The active call object */ protected function _clientStreamRequest( $method, $deserialize, array $metadata = [], array $options = [] ) { $call_factory = $this->_StreamUnaryCallFactory($this->channel); $call = $call_factory($method, $deserialize, $metadata, $options); return $call; } /** * Call a remote method that takes a single argument and returns a stream * of responses. * * @param string $method The name of the method to call * @param mixed $argument The argument to the method * @param callable $deserialize A function that deserializes the responses * @param array $metadata A metadata map to send to the server * (optional) * @param array $options An array of options (optional) * * @return ServerStreamingCall The active call object */ protected function _serverStreamRequest( $method, $argument, $deserialize, array $metadata = [], array $options = [] ) { $call_factory = $this->_UnaryStreamCallFactory($this->channel); $call = $call_factory($method, $argument, $deserialize, $metadata, $options); return $call; } /** * Call a remote method with messages streaming in both directions. * * @param string $method The name of the method to call * @param callable $deserialize A function that deserializes the responses * @param array $metadata A metadata map to send to the server * (optional) * @param array $options An array of options (optional) * * @return BidiStreamingCall The active call object */ protected function _bidiRequest( $method, $deserialize, array $metadata = [], array $options = [] ) { $call_factory = $this->_StreamStreamCallFactory($this->channel); $call = $call_factory($method, $deserialize, $metadata, $options); return $call; } }