php使用weiboad/kafka-php推送错误Could not write xxx bytes to stream解决

线上使用php的客户端是 https://github.com/weiboad/kafka-php 版本:dev-master
线上经常报错误:
Could not write 1422 bytes to stream
追查源码后发现在CommonSocket.php 文件中 writeBlocking方法

 public function writeBlocking(string $buffer): int
    {
        // fwrite to a socket may be partial, so loop until we
        // are done with the entire buffer
        $failedAttempts = 0;
        $bytesWritten   = 0;

        $bytesToWrite = strlen($buffer);

        while ($bytesWritten < $bytesToWrite) {
            // wait for stream to become available for writing
            $writable = $this->select([$this->stream], $this->sendTimeoutSec, $this->sendTimeoutUsec, false);

            if ($writable === false) {
                throw new Exception\Socket('Could not write ' . $bytesToWrite . ' bytes to stream');
            }

            if ($writable === 0) {
                $res = $this->getMetaData();
                if (! empty($res['timed_out'])) {
                    throw new Exception('Timed out writing ' . $bytesToWrite . ' bytes to stream after writing ' . $bytesWritten . ' bytes');
                }

                throw new Exception\Socket('Could not write ' . $bytesToWrite . ' bytes to stream');
            }

            if ($bytesToWrite - $bytesWritten > self::MAX_WRITE_BUFFER) {
                // write max buffer size
                $wrote = fwrite($this->stream, substr($buffer, $bytesWritten, self::MAX_WRITE_BUFFER));
            } else {
                // write remaining buffer bytes to stream
                $wrote = fwrite($this->stream, substr($buffer, $bytesWritten));
            }

            if ($wrote === -1 || $wrote === false) {
                throw new Exception\Socket('Could not write ' . strlen($buffer) . ' bytes to stream, completed writing only ' . $bytesWritten . ' bytes');
            }

            if ($wrote === 0) {
                // Increment the number of times we have failed
                $failedAttempts++;

                if ($failedAttempts > $this->maxWriteAttempts) {
                    throw new Exception\Socket('After ' . $failedAttempts . ' attempts could not write ' . strlen($buffer) . ' bytes to stream, completed writing only ' . $bytesWritten . ' bytes');
                }
            } else {
                // If we wrote something, reset our failed attempt counter
                $failedAttempts = 0;
            }

            $bytesWritten += $wrote;
        }

        return $bytesWritten;
    }

w r i t a b l e = = = f a l s e 这 一 行 抛 出 的 异 常 , 追 查 代 码 发 现 调 用 的 是 s t r e a m s e l e c t ( writable === false 这一行抛出的异常, 追查代码发现调用的是 stream_select( writable===false,streamselect(null, $sockets, $null, $timeoutSec, $timeoutUsec); 方法

stream_select 是一个事件模型, 看文档知道 需要传入$socktes, readSocket, writeSocket, ExceptionSocket,超时秒数 和超时毫秒数 那为什么会返回false 呢
文档解释
https://www.php.net/manual/en/function.stream-select.php

成功时,stream_select()返回已修改数组中包含的流资源的数量,如果超时在任何有趣的事件发生之前到期,则可能为零。FALSE 返回错误并引发警告(如果系统调用被传入信号中断,则可能发生此情况)。

那说明有错误引起的
据需看代码看 到创建socket

 protected function createSocket(string $remoteSocket, $context, ?int &$errno, ?string &$errstr)
    {
        return stream_socket_client(
            $remoteSocket,
            $errno,
            $errstr,
            $this->sendTimeoutSec + ($this->sendTimeoutUsec / 1000000),
            STREAM_CLIENT_CONNECT,
            $context
        );
    }

传入的timeout时间是 t h i s − > s e n d T i m e o u t S e c + ( this->sendTimeoutSec + ( this>sendTimeoutSec+(this->sendTimeoutUsec / 1000000), 计算默认默认值是0.1s
这个值太小了 会导致socket超时直接断开, 在stream_select时候 socket已经断开 返回false

将值改大后实验 不在报错误了 确实是这个值引起的

将源码值改为:

**
     * Send timeout in seconds.
     *
     * @var int
     */
    protected $sendTimeoutSec = 300;

    /**
     * Send timeout in microseconds.
     *
     * @var int
     */
    protected $sendTimeoutUsec = 100000;

    /**
     * Recv timeout in seconds
     *
     * @var int
     */
    protected $recvTimeoutSec = 300;

    /**
     * Recv timeout in microseconds
     *
     * @var int
     */
    protected $recvTimeoutUsec = 750000;

附我的kafka配置:

 $config = \Kafka\ProducerConfig::getInstance();
        $config->setMetadataRefreshIntervalMs(10000);
        $config->setMessageMaxBytes(1000000000);
        $config->setMetadataBrokerList($kafka_config['url']);
        $config->setBrokerVersion('1.1.1');
        $config->setSecurityProtocol($kafka_config['securityprotocol']);
        $config->setSaslMechanism($kafka_config['saslmechanism']);
        $config->setSaslUsername($kafka_config['saslusername']);
        $config->setSaslPassword($kafka_config['saslpassword']);
        $config->setRequiredAck(1);
        $config->setIsAsyn(false);
        $config->setProduceInterval(500);
        $config->setRequestTimeout(900000);
        $config->setTimeout(900000); //kafka 请求超时控制
        $producer = new \Kafka\Producer();
        $producer->setLogger($logger);

默认值都设置最大吧 保证不会出错 哈哈哈

我将项目拉了一份修改后放到了自己的github中
https://github.com/jeevi-cao/kafka-php

码字不易

你可能感兴趣的:(编程语言)