В настоящее время я готов переписать простой код с Python на PHP.Этот код предназначен для запроса к Yandex Cloud (Speech Kit) с использованием ProtoBuf и gRPC.
На Python (v3) все работает просто отлично.
Затем я хочу то же самое в PHP (7.1.23 на MacOS).Я использовал protoc
для компиляции Protobuf, а также плагин grpc php для получения Service Client.
Первое затруднение, с которым я столкнулся, заключается в том, что интерфейс метода клиента SttServiceClient->StreamingRecognize(array metadata[], array options=[])
, что довольно запутанно, так как я предполагал, что он должен принять StreamingRecognitionRequest
.
Во-вторых, я получаю исключение InvalidArgumentException: дано неверное значение метаданных:
PHP Fatal error: Uncaught InvalidArgumentException: Bad metadata value given in /Users/cyberpug/Documents/repos/php/yaskit/vendor/grpc/grpc/src/lib/BidiStreamingCall.php:37
Stack trace:
#0 /Users/cyberpug/Documents/repos/php/yaskit/vendor/grpc/grpc/src/lib/BidiStreamingCall.php(37): Grpc\Call->startBatch(Array)
#1 /Users/cyberpug/Documents/repos/php/yaskit/vendor/grpc/grpc/src/lib/BaseStub.php(384): Grpc\BidiStreamingCall->start(Array)
#2 /Users/cyberpug/Documents/repos/php/yaskit/vendor/grpc/grpc/src/lib/BaseStub.php(595): Grpc\BaseStub->Grpc\{closure}('/yandex.cloud.a...', Array, Array, Array)
#3 /Users/cyberpug/Documents/repos/php/yaskit/STT/proto/Yandex/Cloud/Ai/Stt/V2/SttServiceClient.php(26): Grpc\BaseStub->_bidiRequest('/yandex.cloud.a...', Array, Array, Array)
#4 /Users/cyberpug/Documents/repos/php/yaskit/test.php(83): Yandex\Cloud\Ai\Stt\V2\SttServiceClient->StreamingRecognize(Array, Array)
#5 {main}
thrown in /Users/cyberpug/Documents/repos/php/yaskit/vendor/grpc/grpc/src/lib/BidiStreamingCall.php on line 37
Ну, я здесь бессилен.Я не могу понять, почему это так (не ожидаемый интерфейс клиента) и почему такая ошибка происходит?Сегодня я прочитал так много вещей о Protobuf и gRPC безрезультатно!
Вот код на Python (v3):
class Yandex:
def __init__(self, **kwargs):
self._host = kwargs["url"]
self._credentials = grpc.ssl_channel_credentials()
self._iamToken = kwargs["iamToken"]
spec = stt_service_pb2.RecognitionSpec(
language_code=kwargs["lang"],
profanity_filter=kwargs["profanityFilter"],
model=kwargs["model"],
partial_results=kwargs["partialResults"],
audio_encoding=kwargs["audioEncoding"],
sample_rate_hertz=kwargs["sampleRateHertz"]
)
self._config = stt_service_pb2.RecognitionConfig(
specification=spec,
folder_id=kwargs["folderId"]
)
def _createStub(self):
return stt_service_pb2_grpc.SttServiceStub(
channel=grpc.secure_channel(
self._host,
self._credentials
)
)
def _createHandshake(self):
return stt_service_pb2.StreamingRecognitionRequest(
config=self._config
)
def _createRequest(self, data):
return stt_service_pb2.StreamingRecognitionRequest(
audio_content=data
)
def _wrap(self, method):
yield self._createHandshake()
for chunk in method():
yield self._createRequest(chunk)
def stream(self, method):
stub = self._createStub()
return stub.StreamingRecognize(
self._wrap(method),
metadata=(('authorization', 'Bearer %s' %
self._iamToken),)
)
Вот мой PHP-код, который япробовал:
#!/usr/bin/env php
<?php
namespace yaskit;
require "Configuration.php";
use Yaskit\Settings;
use Yaskit\TokenManager;
use \Yandex\Cloud\Ai\Stt\V2\RecognitionConfig;
use \Yandex\Cloud\Ai\Stt\V2\RecognitionSpec;
use \Yandex\Cloud\Ai\Stt\V2\RecognitionSpec\AudioEncoding;
use \Yandex\Cloud\Ai\Stt\V2\StreamingRecognitionRequest;
use \Yandex\Cloud\Ai\Stt\V2\SttServiceClient;
$auth_set = new Settings();
$app_set = new Settings();
$auth_set->load("settings.auth.json");
$app_set->load("settings.speech-to-text.json");
$mgr = new TokenManager("private-test.pem");
$mgr->useLogger($app->createLogger(TokenManager::class));
$mgr->validate($auth_set);
$url = "stt.api.cloud.yandex.net:443";
$iam = $auth_set["iam-token"];
$raw_req = array(
"config" => array(
"specification" => array(
"sample_rate_hertz" => $app_set["sample-rate-hertz"],
"language_code" => $app_set["lang"],
"profanity_filter" => $app_set["profanity-filter"],
"model" => $app_set["model"],
"partial_results" => $app_set["partial-results"],
"audio_encoding" => AudioEncoding::LINEAR16_PCM,
),
"folder_id" => $auth_set["folder-id"],
),
);
$spec = new RecognitionSpec(array(
"sample_rate_hertz" => $app_set["sample-rate-hertz"],
"language_code" => $app_set["lang"],
"profanity_filter" => $app_set["profanity-filter"],
"model" => $app_set["model"],
"partial_results" => $app_set["partial-results"],
"audio_encoding" => AudioEncoding::LINEAR16_PCM,
));
$config = new RecognitionConfig(array(
"specification" => $spec,
"folder_id" => $auth_set["folder-id"],
));
$req = new StreamingRecognitionRequest(array(
"config" => $config,
));
$service = new SttServiceClient($url, [
"credentials" => \Grpc\ChannelCredentials::createSsl(),
]);
$res = $service->StreamingRecognize(["authorization" => "Bearer $iam"], $raw_req);
Я не знаю, какой из скомпилированных файлов мне следует публиковать здесь (потому что их много), поэтому я публикую только Service Client и сам raw protobuf.Я опубликую больше, если нужно.
SttServiceClient:
<?php
// GENERATED CODE -- DO NOT EDIT!
namespace Yandex\Cloud\Ai\Stt\V2;
/**
*/
class SttServiceClient extends \Grpc\BaseStub {
/**
* @param string $hostname hostname
* @param array $opts channel options
* @param \Grpc\Channel $channel (optional) re-use channel object
*/
public function __construct($hostname, $opts, $channel = null) {
parent::__construct($hostname, $opts, $channel);
}
/**
* @param array $metadata metadata
* @param array $options call options
*/
public function StreamingRecognize($metadata = [], $options = []) {
return $this->_bidiRequest('/yandex.cloud.ai.stt.v2.SttService/StreamingRecognize',
['\Yandex\Cloud\Ai\Stt\V2\StreamingRecognitionResponse','decode'],
$metadata, $options);
}
}
Необработанный протобуф:
syntax = "proto3";
package yandex.cloud.ai.stt.v2;
option go_package = "github.com/yandex-cloud/go-genproto/yandex/cloud/ai/stt/v2;stt";
service SttService {
rpc StreamingRecognize (stream StreamingRecognitionRequest) returns (stream StreamingRecognitionResponse) {
}
}
message StreamingRecognitionRequest {
oneof streaming_request {
RecognitionConfig config = 1;
bytes audio_content = 2;
}
}
message RecognitionConfig {
RecognitionSpec specification = 1;
string folder_id = 2;
}
message RecognitionSpec {
enum AudioEncoding {
AUDIO_ENCODING_UNSPECIFIED = 0;
// 16-bit signed little-endian (Linear PCM)
LINEAR16_PCM = 1;
OGG_OPUS = 2;
}
AudioEncoding audio_encoding = 1;
// 8000, 16000, 48000 only for pcm
int64 sample_rate_hertz = 2;
// code in BCP-47
string language_code = 3;
bool profanity_filter = 4;
string model = 5;
// If set true, tentative hypotheses may be returned as they become available (final=false flag)
// If false or omitted, only final=true result(s) are returned.
bool partial_results = 7;
bool single_utterance = 8;
}
message StreamingRecognitionResponse {
repeated SpeechRecognitionChunk chunks = 1;
bool end_of_single_utterance = 2;
}
message SpeechRecognitionChunk {
repeated SpeechRecognitionAlternative alternatives = 1;
bool final = 2;
}
message SpeechRecognitionAlternative {
string text = 1;
float confidence = 2;
}