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']); } $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'] = ''; } if (!empty($opts['grpc.ssl_target_name_override'])) { $this->hostname_override = $opts['grpc.ssl_target_name_override']; } $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'); } if ($channel) { if (!is_a($channel, 'Grpc\Channel')) { throw new \Exception('The channel argument is not a'. 'Channel object'); } $this->channel = $channel; } else { $this->channel = 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; } /* 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 SimpleSurfaceActiveCall The active call object */ protected function _simpleRequest($method, $argument, $deserialize, array $metadata = [], array $options = []) { $call = new UnaryCall($this->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; } /** * 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 ClientStreamingSurfaceActiveCall The active call object */ protected function _clientStreamRequest($method, $deserialize, array $metadata = [], array $options = []) { $call = new ClientStreamingCall($this->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; } /** * 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 ServerStreamingSurfaceActiveCall The active call object */ protected function _serverStreamRequest($method, $argument, $deserialize, array $metadata = [], array $options = []) { $call = new ServerStreamingCall($this->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; } /** * 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 BidiStreamingSurfaceActiveCall The active call object */ protected function _bidiRequest($method, $deserialize, array $metadata = [], array $options = []) { $call = new BidiStreamingCall($this->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; } }