-
Notifications
You must be signed in to change notification settings - Fork 37
/
RdKafkaConnectionFactory.php
111 lines (95 loc) · 3.23 KB
/
RdKafkaConnectionFactory.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
<?php
declare(strict_types=1);
namespace Enqueue\RdKafka;
use Interop\Queue\ConnectionFactory;
use Interop\Queue\Context;
class RdKafkaConnectionFactory implements ConnectionFactory
{
/**
* @var array
*/
private $config;
/**
* The config could be an array, string DSN or null. In case of null it will attempt to connect to localhost with default settings.
*
* [
* 'global' => [ // https://github.com/edenhill/librdkafka/blob/master/CONFIGURATION.md
* 'metadata.broker.list' => 'localhost:9092',
* ],
* 'topic' => [],
* 'dr_msg_cb' => null,
* 'error_cb' => null,
* 'rebalance_cb' => null,
* 'partitioner' => null, // https://arnaud-lb.github.io/php-rdkafka/phpdoc/rdkafka-topicconf.setpartitioner.html
* 'log_level' => null,
* 'commit_async' => false,
* 'shutdown_timeout' => -1, // https://github.com/arnaud-lb/php-rdkafka#proper-shutdown
* ]
*
* or
*
* kafka://host:port
*
* @param array|string $config
*/
public function __construct($config = 'kafka:')
{
if (version_compare(RdKafkaContext::getLibrdKafkaVersion(), '1.0.0', '<')) {
throw new \RuntimeException('You must install librdkafka:1.0.0 or higher');
}
if (empty($config) || 'kafka:' === $config) {
$config = [];
} elseif (is_string($config)) {
$config = $this->parseDsn($config);
} elseif (is_array($config)) {
} else {
throw new \LogicException('The config must be either an array of options, a DSN string or null');
}
$this->config = array_replace_recursive($this->defaultConfig(), $config);
}
/**
* @return RdKafkaContext
*/
public function createContext(): Context
{
return new RdKafkaContext($this->config);
}
private function parseDsn(string $dsn): array
{
$dsnConfig = parse_url($dsn);
if (false === $dsnConfig) {
throw new \LogicException(sprintf('Failed to parse DSN "%s"', $dsn));
}
$dsnConfig = array_replace([
'scheme' => null,
'host' => null,
'port' => null,
'user' => null,
'pass' => null,
'path' => null,
'query' => null,
], $dsnConfig);
if ('kafka' !== $dsnConfig['scheme']) {
throw new \LogicException(sprintf('The given DSN scheme "%s" is not supported. Could be "kafka" only.', $dsnConfig['scheme']));
}
$config = [];
if ($dsnConfig['query']) {
parse_str($dsnConfig['query'], $config);
}
$broker = $dsnConfig['host'];
if ($dsnConfig['port']) {
$broker .= ':'.$dsnConfig['port'];
}
$config['global']['metadata.broker.list'] = $broker;
return array_replace_recursive($this->defaultConfig(), $config);
}
private function defaultConfig(): array
{
return [
'global' => [
'group.id' => uniqid('', true),
'metadata.broker.list' => 'localhost:9092',
],
];
}
}