Skip to content

Commit fe9b0b9

Browse files
committed
Allow specifying a PSR-3 logger for the server process.
Fix some race conditions on server shutdown and signal handling in child processes.
1 parent 49c35a3 commit fe9b0b9

File tree

7 files changed

+264
-15
lines changed

7 files changed

+264
-15
lines changed

composer.json

+4-3
Original file line numberDiff line numberDiff line change
@@ -18,13 +18,14 @@
1818
"php": ">=7.1",
1919
"freedsx/asn1": "^0.4.0",
2020
"freedsx/socket": "^0.5.2",
21-
"freedsx/sasl": "^0.1.0"
21+
"freedsx/sasl": "^0.1.0",
22+
"psr/log": "^1|^2|^3"
2223
},
2324
"require-dev": {
24-
"phpspec/phpspec": "^5.1|^6.1|^7.1",
25+
"phpspec/phpspec": "^7.2|^7.1|^6.1|^5.1",
2526
"phpunit/phpunit": "^9.3|^8.0|^7.0",
2627
"symplify/easy-coding-standard": "^6.1|^7.3|^9.0",
27-
"friends-of-phpspec/phpspec-code-coverage": "^4.3|^6.1",
28+
"friends-of-phpspec/phpspec-code-coverage": "^4.3|^6.1|dev-master",
2829
"phpstan/phpstan": "^0.12.70",
2930
"symfony/process": "^3.0|^4.0|^5.0",
3031
"squizlabs/php_codesniffer": "3.*",

docs/Server/Configuration.md

+20
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ LDAP Server Configuration
66
* [port](#port)
77
* [unix_socket](#unix_socket)
88
* [transport](#transport)
9+
* [logger](#logger)
910
* [idle_timeout](#idle_timeout)
1011
* [require_authentication](#require_authentication)
1112
* [allow_anonymous](#allow_anonymous)
@@ -75,6 +76,25 @@ If using `unix` for the transport you can change set the `unix_socket` to a file
7576

7677
**Default**: `tcp`
7778

79+
------------------
80+
#### logger
81+
82+
Specify a PSR-3 compatible logging instance to use. This will log various server events and errors.
83+
84+
You can also set the logger after instantiating the server and before running it:
85+
86+
```php
87+
use FreeDSx\Ldap\LdapServer;
88+
89+
$server = new LdapServer();
90+
91+
// instantiate some logger class...
92+
93+
$server->useLogger($logger);
94+
```
95+
96+
**Default**: `null`
97+
7898
------------------
7999
#### idle_timeout
80100

src/FreeDSx/Ldap/LdapServer.php

+16-2
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,7 @@
2121
use FreeDSx\Ldap\Server\ServerRunner\ServerRunnerInterface;
2222
use FreeDSx\Socket\Exception\ConnectionException;
2323
use FreeDSx\Socket\SocketServer;
24+
use Psr\Log\LoggerInterface;
2425

2526
/**
2627
* The LDAP server.
@@ -29,6 +30,8 @@
2930
*/
3031
class LdapServer
3132
{
33+
use LoggerTrait;
34+
3235
/**
3336
* @var array
3437
*/
@@ -43,6 +46,7 @@ class LdapServer
4346
'request_handler' => null,
4447
'rootdse_handler' => null,
4548
'paging_handler' => null,
49+
'logger' => null,
4650
'use_ssl' => false,
4751
'ssl_cert' => null,
4852
'ssl_cert_passphrase' => null,
@@ -147,6 +151,16 @@ public function usePagingHandler(PagingHandlerInterface $pagingHandler): self
147151
return $this;
148152
}
149153

154+
/**
155+
* Specify a logger to be used by the server process.
156+
*/
157+
public function useLogger(LoggerInterface $logger): self
158+
{
159+
$this->options['logger'] = $logger;
160+
161+
return $this;
162+
}
163+
150164
/**
151165
* Convenience method for generating an LDAP server instance that will proxy client request's to an LDAP server.
152166
*
@@ -191,14 +205,14 @@ private function removeExistingSocketIfNeeded(string $socket): void
191205
}
192206

193207
if (!is_writeable($socket)) {
194-
throw new RuntimeException(sprintf(
208+
$this->logAndThrow(sprintf(
195209
'The socket "%s" already exists and is not writeable. To run the LDAP server, you must remove the existing socket.',
196210
$socket
197211
));
198212
}
199213

200214
if (!unlink($socket)) {
201-
throw new RuntimeException(sprintf(
215+
$this->logAndThrow(sprintf(
202216
'The existing socket "%s" could not be removed. To run the LDAP server, you must remove the existing socket.',
203217
$socket
204218
));

src/FreeDSx/Ldap/LoggerTrait.php

+85
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,85 @@
1+
<?php
2+
3+
/**
4+
* This file is part of the FreeDSx LDAP package.
5+
*
6+
* (c) Chad Sikorra <[email protected]>
7+
*
8+
* For the full copyright and license information, please view the LICENSE
9+
* file that was distributed with this source code.
10+
*/
11+
12+
namespace FreeDSx\Ldap;
13+
14+
use FreeDSx\Ldap\Exception\RuntimeException;
15+
use Psr\Log\LoggerInterface;
16+
use Psr\Log\LogLevel;
17+
18+
/**
19+
* Some simple logging methods. Only logs if we have a logger in the options.
20+
*
21+
* @author Chad Sikorra <[email protected]>
22+
*/
23+
trait LoggerTrait
24+
{
25+
/**
26+
* Logs a message and then throws a runtime exception.
27+
*
28+
* @throws RuntimeException
29+
*/
30+
private function logAndThrow(
31+
string $message,
32+
array $context = []
33+
): void {
34+
$this->log(
35+
LogLevel::ERROR,
36+
$message,
37+
$context
38+
);
39+
40+
throw new RuntimeException($message);
41+
}
42+
43+
protected function logError(
44+
string $message,
45+
array $context = []
46+
): void {
47+
$this->log(
48+
LogLevel::ERROR,
49+
$message,
50+
$context
51+
);
52+
}
53+
54+
protected function logInfo(
55+
string $message,
56+
array $context = []
57+
): void {
58+
$this->log(
59+
LogLevel::INFO,
60+
$message,
61+
$context
62+
);
63+
}
64+
65+
/**
66+
* Log a message with a level and context (if we have a logger).
67+
*
68+
* @param string $level
69+
* @param string $message
70+
* @param array $context
71+
*/
72+
private function log(
73+
string $level,
74+
string $message,
75+
array $context = []
76+
): void {
77+
if (isset($this->options['logger']) && $this->options['logger'] instanceof LoggerInterface) {
78+
$this->options['logger']->log(
79+
$level,
80+
$message,
81+
$context
82+
);
83+
}
84+
}
85+
}

src/FreeDSx/Ldap/Protocol/ServerProtocolHandler.php

+26-2
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424
use FreeDSx\Ldap\Protocol\Queue\ServerQueue;
2525
use FreeDSx\Ldap\Server\HandlerFactoryInterface;
2626
use FreeDSx\Ldap\Server\RequestHistory;
27+
use FreeDSx\Ldap\LoggerTrait;
2728
use FreeDSx\Ldap\Server\Token\TokenInterface;
2829
use FreeDSx\Socket\Exception\ConnectionException;
2930
use Throwable;
@@ -37,6 +38,8 @@
3738
*/
3839
class ServerProtocolHandler
3940
{
41+
use LoggerTrait;
42+
4043
/**
4144
* @var array
4245
*/
@@ -85,6 +88,11 @@ class ServerProtocolHandler
8588
*/
8689
protected $bindHandlerFactory;
8790

91+
/**
92+
* @var array<string, mixed>
93+
*/
94+
protected $defaultContext = [];
95+
8896
public function __construct(
8997
ServerQueue $queue,
9098
HandlerFactoryInterface $handlerFactory,
@@ -111,9 +119,10 @@ public function __construct(
111119
*
112120
* @throws EncoderException
113121
*/
114-
public function handle(): void
122+
public function handle(array $defaultContext = []): void
115123
{
116124
$message = null;
125+
$this->defaultContext = $defaultContext;
117126

118127
try {
119128
while ($message = $this->queue->getMessage()) {
@@ -135,7 +144,18 @@ public function handle(): void
135144
# Per RFC 4511, 4.1.1 if the PDU cannot be parsed or is otherwise malformed a disconnect should be sent with a
136145
# result code of protocol error.
137146
$this->sendNoticeOfDisconnect('The message encoding is malformed.');
147+
$this->logError(
148+
'The client sent a malformed request. Terminating their connection.',
149+
$this->defaultContext
150+
);
138151
} catch (Exception | Throwable $e) {
152+
$this->logError(
153+
'An unexpected exception was caught while handling the client. Terminating their connection.',
154+
array_merge(
155+
$this->defaultContext,
156+
['exception' => $e]
157+
)
158+
);
139159
if ($this->queue->isConnected()) {
140160
$this->sendNoticeOfDisconnect();
141161
}
@@ -151,13 +171,17 @@ public function handle(): void
151171
*
152172
* @throws EncoderException
153173
*/
154-
public function shutdown(): void
174+
public function shutdown(array $context = []): void
155175
{
156176
$this->sendNoticeOfDisconnect(
157177
'The server is shutting down.',
158178
ResultCode::UNAVAILABLE
159179
);
160180
$this->queue->close();
181+
$this->logInfo(
182+
'Sent notice of disconnect to client and closed the connection.',
183+
$context
184+
);
161185
}
162186

163187
/**

0 commit comments

Comments
 (0)