Skip to content

Commit 846f0de

Browse files
authoredDec 9, 2022
Add connection setting for blocking mode of socket (#135)
* Add connection setting for blocking mode of socket * Add warning that blocking socket is potentially dangerous
1 parent a0f8186 commit 846f0de

File tree

4 files changed

+70
-11
lines changed

4 files changed

+70
-11
lines changed
 

‎README.md

+9
Original file line numberDiff line numberDiff line change
@@ -142,6 +142,15 @@ $connectionSettings = (new \PhpMqtt\Client\ConnectionSettings)
142142
// The password used for authentication when connecting to the broker.
143143
->setPassword(null)
144144

145+
// Whether to use a blocking socket or not. By default, the socket is non-blocking,
146+
// which is required when using subscriptions and/or {@see MqttClient::loop()}.
147+
// In rare cases, it might be required to use a blocking socket though. One such example
148+
// is when sending large messages (e.g. binaries) and the broker has a limited receive buffer.
149+
//
150+
// Note: When using a blocking socket, the MQTT client can get stuck if the socket is broken
151+
// or when the broker does not consume the sent data fast enough. Use with caution.
152+
->useBlockingSocket(false)
153+
145154
// The connect timeout defines the maximum amount of seconds the client will try to establish
146155
// a socket connection with the broker. The value cannot be less than 1 second.
147156
->setConnectTimeout(60)

‎src/ConnectionSettings.php

+30
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ class ConnectionSettings
1313
{
1414
private ?string $username = null;
1515
private ?string $password = null;
16+
private bool $useBlockingSocket = false;
1617
private int $connectTimeout = 60;
1718
private int $socketTimeout = 5;
1819
private int $resendTimeout = 10;
@@ -80,6 +81,35 @@ public function getPassword(): ?string
8081
return $this->password;
8182
}
8283

84+
/**
85+
* Whether to use a blocking socket or not. By default, the socket is non-blocking,
86+
* which is required when using subscriptions and/or {@see MqttClient::loop()}.
87+
* In rare cases, it might be required to use a blocking socket though. One such example
88+
* is when sending large messages (e.g. binaries) and the broker has a limited receive buffer.
89+
*
90+
* Note: When using a blocking socket, the MQTT client can get stuck if the socket is broken
91+
* or when the broker does not consume the sent data fast enough. Use with caution.
92+
*
93+
* @param bool $useBlockingSocket
94+
* @return ConnectionSettings
95+
*/
96+
public function useBlockingSocket(bool $useBlockingSocket): ConnectionSettings
97+
{
98+
$copy = clone $this;
99+
100+
$copy->useBlockingModeForSocket = $useBlockingSocket;
101+
102+
return $copy;
103+
}
104+
105+
/**
106+
* @return bool
107+
*/
108+
public function shouldUseBlockingSocket(): bool
109+
{
110+
return $this->useBlockingSocket;
111+
}
112+
83113
/**
84114
* The connect timeout is the maximum amount of seconds the client will try to establish
85115
* a socket connection with the broker. The value cannot be less than 1 second.

‎src/MqttClient.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -290,7 +290,7 @@ protected function establishSocketConnection(): void
290290
}
291291

292292
stream_set_timeout($socket, $this->settings->getSocketTimeout());
293-
stream_set_blocking($socket, false);
293+
stream_set_blocking($socket, $this->settings->shouldUseBlockingSocket());
294294

295295
$this->logger->debug('Socket opened and ready to use.');
296296

‎tests/Feature/PublishSubscribeTest.php

+30-10
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77

88
namespace Tests\Feature;
99

10+
use PhpMqtt\Client\ConnectionSettings;
1011
use PhpMqtt\Client\MqttClient;
1112
use Tests\TestCase;
1213

@@ -20,29 +21,40 @@ class PublishSubscribeTest extends TestCase
2021
public function publishSubscribeData(): array
2122
{
2223
return [
23-
['test/foo/bar/baz', 'test/foo/bar/baz', 'hello world', []],
24-
['test/foo/bar/+', 'test/foo/bar/baz', 'hello world', ['baz']],
25-
['test/foo/+/baz', 'test/foo/bar/baz', 'hello world', ['bar']],
26-
['test/foo/#', 'test/foo/bar/baz', 'hello world', ['bar/baz']],
27-
['test/foo/+/bar/#', 'test/foo/my/bar/baz', 'hello world', ['my', 'baz']],
28-
['test/foo/+/bar/#', 'test/foo/my/bar/baz/blub', 'hello world', ['my', 'baz/blub']],
29-
['test/foo/bar/baz', 'test/foo/bar/baz', random_bytes(2 * 1024 * 1024), []], // 2MB message
24+
[false, 'test/foo/bar/baz', 'test/foo/bar/baz', 'hello world', []],
25+
[false, 'test/foo/bar/+', 'test/foo/bar/baz', 'hello world', ['baz']],
26+
[false, 'test/foo/+/baz', 'test/foo/bar/baz', 'hello world', ['bar']],
27+
[false, 'test/foo/#', 'test/foo/bar/baz', 'hello world', ['bar/baz']],
28+
[false, 'test/foo/+/bar/#', 'test/foo/my/bar/baz', 'hello world', ['my', 'baz']],
29+
[false, 'test/foo/+/bar/#', 'test/foo/my/bar/baz/blub', 'hello world', ['my', 'baz/blub']],
30+
[false, 'test/foo/bar/baz', 'test/foo/bar/baz', random_bytes(2 * 1024 * 1024), []], // 2MB message
31+
[true, 'test/foo/bar/baz', 'test/foo/bar/baz', 'hello world', []],
32+
[true, 'test/foo/bar/+', 'test/foo/bar/baz', 'hello world', ['baz']],
33+
[true, 'test/foo/+/baz', 'test/foo/bar/baz', 'hello world', ['bar']],
34+
[true, 'test/foo/#', 'test/foo/bar/baz', 'hello world', ['bar/baz']],
35+
[true, 'test/foo/+/bar/#', 'test/foo/my/bar/baz', 'hello world', ['my', 'baz']],
36+
[true, 'test/foo/+/bar/#', 'test/foo/my/bar/baz/blub', 'hello world', ['my', 'baz/blub']],
37+
[true, 'test/foo/bar/baz', 'test/foo/bar/baz', random_bytes(2 * 1024 * 1024), []], // 2MB message
3038
];
3139
}
3240

3341
/**
3442
* @dataProvider publishSubscribeData
3543
*/
3644
public function test_publishing_and_subscribing_using_quality_of_service_0_works_as_intended(
45+
bool $useBlockingSocket,
3746
string $subscriptionTopicFilter,
3847
string $publishTopic,
3948
string $publishMessage,
4049
array $matchedTopicWildcards
4150
): void
4251
{
4352
// We connect and subscribe to a topic using the first client.
53+
$connectionSettings = (new ConnectionSettings())
54+
->useBlockingSocket($useBlockingSocket);
55+
4456
$subscriber = new MqttClient($this->mqttBrokerHost, $this->mqttBrokerPort, 'subscriber');
45-
$subscriber->connect(null, true);
57+
$subscriber->connect($connectionSettings, true);
4658

4759
$subscriber->subscribe(
4860
$subscriptionTopicFilter,
@@ -76,15 +88,19 @@ function (string $topic, string $message, bool $retained, array $wildcards) use
7688
* @dataProvider publishSubscribeData
7789
*/
7890
public function test_publishing_and_subscribing_using_quality_of_service_1_works_as_intended(
91+
bool $useBlockingSocket,
7992
string $subscriptionTopicFilter,
8093
string $publishTopic,
8194
string $publishMessage,
8295
array $matchedTopicWildcards
8396
): void
8497
{
8598
// We connect and subscribe to a topic using the first client.
99+
$connectionSettings = (new ConnectionSettings())
100+
->useBlockingSocket($useBlockingSocket);
101+
86102
$subscriber = new MqttClient($this->mqttBrokerHost, $this->mqttBrokerPort, 'subscriber');
87-
$subscriber->connect(null, true);
103+
$subscriber->connect($connectionSettings, true);
88104

89105
$subscriber->subscribe(
90106
$subscriptionTopicFilter,
@@ -118,15 +134,19 @@ function (string $topic, string $message, bool $retained, array $wildcards) use
118134
* @dataProvider publishSubscribeData
119135
*/
120136
public function test_publishing_and_subscribing_using_quality_of_service_2_works_as_intended(
137+
bool $useBlockingSocket,
121138
string $subscriptionTopicFilter,
122139
string $publishTopic,
123140
string $publishMessage,
124141
array $matchedTopicWildcards
125142
): void
126143
{
127144
// We connect and subscribe to a topic using the first client.
145+
$connectionSettings = (new ConnectionSettings())
146+
->useBlockingSocket($useBlockingSocket);
147+
128148
$subscriber = new MqttClient($this->mqttBrokerHost, $this->mqttBrokerPort, 'subscriber');
129-
$subscriber->connect(null, true);
149+
$subscriber->connect($connectionSettings, true);
130150

131151
$subscription = function (string $topic, string $message, bool $retained, array $wildcards) use ($subscriber, $subscriptionTopicFilter, $publishTopic, $publishMessage, $matchedTopicWildcards) {
132152
// By asserting something here, we will avoid a no-assertions-in-test warning, making the test pass.

0 commit comments

Comments
 (0)
Please sign in to comment.