Skip to content

Commit 638c5dd

Browse files
committed
[1.x] Ensure connection close handler is cleaned up for each request
This changeset resolves a small memory leak that causes roughly 1KB per connection tops. Which isn't a big issue but will make memory fluctuate more. The changeset doesn't introduce any performance degradation. Resolves: #514 Builds on top of: #405, #467, and many others
1 parent 706edec commit 638c5dd

File tree

2 files changed

+45
-10
lines changed

2 files changed

+45
-10
lines changed

src/Io/StreamingServer.php

+10-3
Original file line numberDiff line numberDiff line change
@@ -157,10 +157,17 @@ public function handleRequest(ConnectionInterface $conn, ServerRequestInterface
157157
}
158158

159159
// cancel pending promise once connection closes
160+
$connectionOnCloseResponseCancelerHandler = function () {};
160161
if ($response instanceof PromiseInterface && \method_exists($response, 'cancel')) {
161-
$conn->on('close', function () use ($response) {
162+
$connectionOnCloseResponseCanceler = function () use ($response) {
162163
$response->cancel();
163-
});
164+
};
165+
$connectionOnCloseResponseCancelerHandler = function () use ($connectionOnCloseResponseCanceler, $conn) {
166+
if ($connectionOnCloseResponseCanceler !== null) {
167+
$conn->removeListener('close', $connectionOnCloseResponseCanceler);
168+
}
169+
};
170+
$conn->on('close', $connectionOnCloseResponseCanceler);
164171
}
165172

166173
// happy path: response returned, handle and return immediately
@@ -201,7 +208,7 @@ function ($error) use ($that, $conn, $request) {
201208
$that->emit('error', array($exception));
202209
return $that->writeError($conn, Response::STATUS_INTERNAL_SERVER_ERROR, $request);
203210
}
204-
);
211+
)->then($connectionOnCloseResponseCancelerHandler, $connectionOnCloseResponseCancelerHandler);
205212
}
206213

207214
/** @internal */

tests/Io/StreamingServerTest.php

+35-7
Original file line numberDiff line numberDiff line change
@@ -25,9 +25,17 @@ class StreamingServerTest extends TestCase
2525
*/
2626
public function setUpConnectionMockAndSocket()
2727
{
28-
$this->connection = $this->getMockBuilder('React\Socket\Connection')
28+
$this->connection = $this->mockConnection();
29+
30+
$this->socket = new SocketServerStub();
31+
}
32+
33+
34+
private function mockConnection(array $additionalMethods = null)
35+
{
36+
$connection = $this->getMockBuilder('React\Socket\Connection')
2937
->disableOriginalConstructor()
30-
->setMethods(
38+
->setMethods(array_merge(
3139
array(
3240
'write',
3341
'end',
@@ -39,14 +47,15 @@ public function setUpConnectionMockAndSocket()
3947
'getRemoteAddress',
4048
'getLocalAddress',
4149
'pipe'
42-
)
43-
)
50+
),
51+
(is_array($additionalMethods) ? $additionalMethods : array())
52+
))
4453
->getMock();
4554

46-
$this->connection->method('isWritable')->willReturn(true);
47-
$this->connection->method('isReadable')->willReturn(true);
55+
$connection->method('isWritable')->willReturn(true);
56+
$connection->method('isReadable')->willReturn(true);
4857

49-
$this->socket = new SocketServerStub();
58+
return $connection;
5059
}
5160

5261
public function testRequestEventWillNotBeEmittedForIncompleteHeaders()
@@ -3245,6 +3254,25 @@ public function testNewConnectionWillInvokeParserTwiceAfterInvokingRequestHandle
32453254
$this->assertCount(1, $this->connection->listeners('close'));
32463255
}
32473256

3257+
public function testCompletingARequestWillRemoveConnectionOnCloseListener()
3258+
{
3259+
$connection = $this->mockConnection(array('removeListener'));
3260+
3261+
$request = new ServerRequest('GET', 'http://localhost/');
3262+
3263+
$server = new StreamingServer(Loop::get(), function () {
3264+
return \React\Promise\resolve(new Response());
3265+
});
3266+
3267+
$server->listen($this->socket);
3268+
$this->socket->emit('connection', array($connection));
3269+
3270+
$connection->expects($this->once())->method('removeListener');
3271+
3272+
// pretend parser just finished parsing
3273+
$server->handleRequest($connection, $request);
3274+
}
3275+
32483276
private function createGetRequest()
32493277
{
32503278
$data = "GET / HTTP/1.1\r\n";

0 commit comments

Comments
 (0)