Test Setup Failed
Push — testing ( 3426ad...e1d3e5 )
by Roman
02:50
created

Connection::setAuthCredentials()   A

Complexity

Conditions 3
Paths 2

Size

Total Lines 7

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 2
CRAP Score 3

Importance

Changes 0
Metric Value
dl 0
loc 7
ccs 2
cts 2
cp 1
rs 10
c 0
b 0
f 0
cc 3
nc 2
nop 1
crap 3
1
<?php
2
3
namespace lroman242\LaravelCassandra;
4
5
use Cassandra;
6
use Cassandra\BatchStatement;
7
use Cassandra\Cluster\Builder as ClusterBuilder;
8
use \lroman242\LaravelCassandra\Exceptions\CassandraNotSupportedException;
9
10
class Connection extends \Illuminate\Database\Connection
11
{
12
    const DEFAULT_PAGE_SIZE = 5000;
13
14
    /**
15
     * The Cassandra keyspace
16
     *
17
     * @var string
18
     */
19
    protected $keyspace;
20
21
    /**
22
     * The Cassandra cluster
23
     *
24
     * @var \Cassandra\Cluster
25
     */
26
    protected $cluster;
27
28
    /**
29
     * The Cassandra connection handler.
30
     *
31
     * @var \Cassandra\Session
32
     */
33
    protected $session;
34
35
    /**
36
     * The config
37
     *
38
     * @var array
39
     */
40
    protected $config;
41
42
    /**
43
     * Create a new database connection instance.
44
     *
45
     * @param array $config
46 48
     */
47
    public function __construct(array $config)
48 48
    {
49 48
        $this->config = $config;
50 48
        if (empty($this->config['page_size'])) {
51
            $this->config['page_size'] = self::DEFAULT_PAGE_SIZE;
52
        }
53
54 48
        // Create the connection
55
        $this->cluster = $this->createCluster();
56 48
57 48
        if (isset($config['keyspace'])) {
58
            $keyspaceName = $config['keyspace'];
59 48
60 48
            $this->keyspace = $keyspaceName;
61
            $this->session = $this->cluster->connect($keyspaceName);
62
        }
63 48
64
        $this->useDefaultPostProcessor();
65 48
66
        $this->useDefaultSchemaGrammar();
67 48
68 48
        $this->setQueryGrammar($this->getDefaultQueryGrammar());
69
    }
70
71
    /**
72
     * Begin a fluent query against a database table.
73
     *
74
     * @param string $table
75
     * @return Query\Builder
76 5
     */
77
    public function table($table)
78 5
    {
79
        $processor = $this->getPostProcessor();
80 5
81
        $query = new Query\Builder($this, null, $processor);
0 ignored issues
show
Documentation introduced by
$processor is of type object<Illuminate\Databa...y\Processors\Processor>, but the function expects a null|object<lroman242\La...sandra\Query\Processor>.

It seems like the type of the argument is not accepted by the function/method which you are calling.

In some cases, in particular if PHP’s automatic type-juggling kicks in this might be fine. In other cases, however this might be a bug.

We suggest to add an explicit type cast like in the following example:

function acceptsInteger($int) { }

$x = '123'; // string "123"

// Instead of
acceptsInteger($x);

// we recommend to use
acceptsInteger((integer) $x);
Loading history...
82 5
83
        return $query->from($table);
84
    }
85
86
    /**
87
     * return Cassandra cluster.
88
     *
89
     * @return \Cassandra\Cluster
90 1
     */
91
    public function getCassandraCluster()
92 1
    {
93
        return $this->cluster;
94
    }
95
96
    /**
97
     * return Cassandra Session.
98
     *
99
     * @return \Cassandra\Session
100 3
     */
101
    public function getCassandraSession()
102 3
    {
103
        return $this->session;
104
    }
105
106
    /**
107
     * Return the Cassandra keyspace
108
     *
109
     * @return string
110 1
     */
111
    public function getKeyspace()
112 1
    {
113
        return $this->keyspace;
114
    }
115
116
    /**
117
     * Set username and password to cluster connection.
118
     * Set cluster contact points (IP addresses)
119
     * Set connection communication port
120
     *
121
     * @param \Cassandra\Cluster\Builder $cluster
122 48
     */
123
    protected function setConnectionOptions(ClusterBuilder $cluster)
124 48
    {
125
        // Authentication
126
        if (isset($this->config['username']) && isset($this->config['password'])) {
127 48
            $cluster->withCredentials($this->config['username'], $this->config['password']);
128 48
        }
129
130
        if (!empty($this->config['host'])) {
131
            $contactPoints = $this->config['host'];
132 48
133 48
            if (is_string($contactPoints)) {
134
                $contactPoints = explode(',', $contactPoints);
135 48
            }
136 48
137
            call_user_func_array([$cluster, 'withContactPoints'], (array)$contactPoints);
138
        }
139 48
140
        if (!empty($this->config['port'])) {
141
            $cluster->withPort((int)$this->config['port']);
142 48
        }
143 48
    }
144
145
    /**
146 48
     * Set default consistency level
147
     * Set default timeouts to queries
148 48
     * Set default response size to queries
149 48
     *
150
     * @param \Cassandra\Cluster\Builder $cluster
151
     */
152
    protected function setDefaultQueryOptions(ClusterBuilder $cluster)
153
    {
154
        if (isset($this->config['consistency']) && in_array($this->config['consistency'], [
155 48
                Cassandra::CONSISTENCY_ANY, Cassandra::CONSISTENCY_ONE, Cassandra::CONSISTENCY_TWO,
156
                Cassandra::CONSISTENCY_THREE, Cassandra::CONSISTENCY_QUORUM, Cassandra::CONSISTENCY_ALL,
157
                Cassandra::CONSISTENCY_SERIAL, Cassandra::CONSISTENCY_QUORUM, Cassandra::CONSISTENCY_LOCAL_QUORUM,
158 48
                Cassandra::CONSISTENCY_EACH_QUORUM, Cassandra::CONSISTENCY_LOCAL_SERIAL, Cassandra::CONSISTENCY_LOCAL_ONE,
159 48
            ])) {
160
161
            $cluster->withDefaultConsistency($this->config['consistency']);
162 48
        }
163 48
164
        if (!empty($this->config['timeout'])) {
165
            $cluster->withDefaultTimeout(intval($this->config['timeout']));
166 48
        }
167 48
168
        if (!empty($this->config['connect_timeout'])) {
169
            $cluster->withConnectTimeout(floatval($this->config['connect_timeout']));
170 48
        }
171
172
        if (!empty($this->config['request_timeout'])) {
173
            $cluster->withRequestTimeout(floatval($this->config['request_timeout']));
174
        }
175
176 3
        $cluster->withDefaultPageSize(intval(!empty($this->config['page_size']) ? $this->config['page_size'] : self::DEFAULT_PAGE_SIZE));
177
    }
178 3
179 3
    /**
180 3
     * Create a new Cassandra cluster object.
181
     *
182
     * @return \Cassandra\Cluster
183
     */
184
    protected function createCluster()
185
    {
186
        $cluster = Cassandra::cluster();
187 1
188
        // Authentication
189 1
        $this->setConnectionOptions($cluster);
190
191
        $this->setDefaultQueryOptions($cluster);
192
193
        return $cluster->build();
194
    }
195
196
    /**
197
     * Disconnect from the underlying Cassandra connection.
198
     */
199
    public function disconnect()
200
    {
201
        $this->session->close();
202 48
        $this->session = null;
203
    }
204 48
205
    /**
206
     * Get the PDO driver name.
207
     *
208
     * @return string
209
     */
210
    public function getDriverName()
211
    {
212
        return 'cassandra';
213
    }
214
215
    /**
216
     * Run a select statement against the database.
217 6
     *
218
     * @param  string  $query
219 6
     * @param  array  $bindings
220
     * @param  bool  $useReadPdo
221
     * @param  array  $customOptions
222
     *
223
     * @return mixed
224
     */
225
    public function select($query, $bindings = [], $useReadPdo = true, array $customOptions = [])
226
    {
227
        return $this->statement($query, $bindings, $customOptions);
228
    }
229
230
    /**
231
     * Run an bulk insert statement against the database.
232
     *
233
     * @param  array  $queries
234 7
     * @param  array  $bindings
235 7
     * @param  int  $type
236 1
     * @param  array  $customOptions
237
     *
238
     * @return bool
239 6
     */
240
    public function insertBulk($queries = [], $bindings = [], $type = Cassandra::BATCH_LOGGED, array $customOptions = [])
241 6
    {
242 6
        return $this->batchStatement($queries, $bindings, $type, $customOptions);
243 6
    }
244
245
    /**
246 6
     * Execute a group of queries inside a batch statement against the database.
247 7
     *
248
     * @param  array  $queries
249
     * @param  array  $bindings
250
     * @param  int  $type
251
     * @param  array  $customOptions
252
     *
253
     * @return bool
254
     */
255
    public function batchStatement($queries = [], $bindings = [], $type = Cassandra::BATCH_LOGGED, array $customOptions = [])
256
    {
257
        return $this->run($queries, $bindings, function ($queries, $bindings) use ($type, $customOptions) {
0 ignored issues
show
Documentation introduced by
$queries is of type array, but the function expects a string.

It seems like the type of the argument is not accepted by the function/method which you are calling.

In some cases, in particular if PHP’s automatic type-juggling kicks in this might be fine. In other cases, however this might be a bug.

We suggest to add an explicit type cast like in the following example:

function acceptsInteger($int) { }

$x = '123'; // string "123"

// Instead of
acceptsInteger($x);

// we recommend to use
acceptsInteger((integer) $x);
Loading history...
258
            if ($this->pretending()) {
259 48
                return [];
260
            }
261 48
262
            $batch = new BatchStatement($type);
263
264
            foreach ($queries as $k => $query) {
265
                $preparedStatement = $this->session->prepare($query);
266
                $batch->add($preparedStatement, $bindings[$k]);
267
            }
268
269
            return $this->session->execute($batch, $customOptions);
270
        });
271
    }
272
273
    /**
274
     * Execute an CQL statement and return the boolean result.
275 10
     *
276
     * @param  string  $query
277 10
     * @param  array   $bindings
278
     * @param  array  $customOptions
279
     *
280
     * @return mixed
281
     */
282
    public function statement($query, $bindings = [], array $customOptions = [])
283 48
    {
284
        return $this->runStatement($query, $bindings, $customOptions);
285 48
    }
286
287
    /**
288
     * Because Cassandra is an eventually consistent database, it's not possible to obtain
289
     * the affected count for statements so we're just going to return 0, based on the idea
290
     * that if the query fails somehow, an exception will be thrown
291 48
     *
292
     * @param  string  $query
293 48
     * @param  array   $bindings
294
     * @param  array  $customOptions
295
     *
296
     * @return int
297
     */
298
    public function affectingStatement($query, $bindings = [], array $customOptions = [])
299 48
    {
300
        return $this->runStatement($query, $bindings, $customOptions, 0, 1);
301
    }
302 48
303
    /**
304
     * @inheritdoc
305
     */
306
    protected function getDefaultPostProcessor()
307
    {
308
        return new Query\Processor();
309 48
    }
310
311 48
    /**
312 1
     * @inheritdoc
313
     */
314 48
    protected function getDefaultQueryGrammar()
315
    {
316
        return new Query\Grammar();
317
    }
318
319
    /**
320
     * @inheritdoc
321
     */
322
    protected function getDefaultSchemaGrammar()
323 1
    {
324
        //return new Schema\Grammar();
325 1
    }
326
327
    /**
328
     * Reconnect to the database if connection is missing.
329
     *
330
     * @return void
331
     */
332
    protected function reconnectIfMissingConnection()
333
    {
334
        if (is_null($this->session)) {
335
            $this->session = $this->createCluster()->connect($this->keyspace);
336
        }
337
    }
338
339
    /**
340
     * Dynamically pass methods to the connection.
341 48
     *
342 48
     * @param  string  $method
343 2
     * @param  array   $parameters
344
     * @return mixed
345
     */
346 48
    public function __call($method, $parameters)
347
    {
348
        return call_user_func_array([$this->session, $method], $parameters);
349 48
    }
350
351 48
    /**
352
     * Execute an CQL statement and return the boolean result.
353 48
     *
354 48
     * @param string $query
355
     * @param array $bindings
356
     * @param array $customOptions
357
     * @param mixed $defaultFailed
358
     * @param mixed $defaultSuccess
359
     *
360 1
     * @return mixed
361
     */
362 1
    protected function runStatement($query, $bindings = [], array $customOptions = [], $defaultFailed = [], $defaultSuccess = null)
363
    {
364
        return $this->run($query, $bindings, function ($query, $bindings) use ($customOptions, $defaultFailed, $defaultSuccess) {
365
            if ($this->pretending()) {
366
                return $defaultFailed;
367
            }
368 1
369
            $preparedStatement = $this->session->prepare($query);
370 1
371
            //Add bindings
372
            $customOptions['arguments'] = $bindings;
373
374
            $result = $this->session->execute($preparedStatement, $customOptions);
375
376 1
            return $defaultSuccess === null ? $result : $defaultSuccess;
377
        });
378 1
    }
379
380
    /**
381
     * @inheritDoc
382
     */
383
    public function transaction(\Closure $callback, $attempts = 1)
384 1
    {
385
        throw new CassandraNotSupportedException("Transactions is not supported by Cassandra database");
386 1
    }
387
388
    /**
389
     * @inheritDoc
390
     */
391
    public function beginTransaction()
392 1
    {
393
        throw new CassandraNotSupportedException("Transactions is not supported by Cassandra database");
394 1
    }
395
396
    /**
397
     * @inheritDoc
398
     */
399
    public function commit()
400
    {
401
        throw new CassandraNotSupportedException("Transactions is not supported by Cassandra database");
402
    }
403
404
    /**
405
     * @inheritDoc
406
     */
407
    public function rollBack()
408
    {
409
        throw new CassandraNotSupportedException("Transactions is not supported by Cassandra database");
410
    }
411
412
    /**
413
     * @inheritDoc
414
     */
415
    public function transactionLevel()
416
    {
417
        throw new CassandraNotSupportedException("Transactions is not supported by Cassandra database");
418
    }
419
420
    //TODO: override isDoctrineAvailable method
421
    //TODO: override getDoctrineColumn method
422
    //TODO: override getDoctrineSchemaManager method
423
    //TODO: override getDoctrineConnection method
424
    //TODO: override getPdo method
425
    //TODO: override getReadPdo method
426
    //TODO: override setPdo method
427
    //TODO: override setReadPdo method
428
    //TODO: override setReconnector method
429
    //TODO: override reconnect method
430
    //TODO: override query method
431
432
    //TODO: override bindValues method
433
    //TODO: override cursor method
434
    //TODO: override unprepared method
435
436
    //TODO: check prepareBindings method
437
438
    //TODO: provide interface for $this->session->executeAsync
439
    //TODO: provide interface for $this->session->prepareAsync
440
    //TODO: provide interface for $this->session->closeAsync
441
    //TODO: provide interface for $this->session->schema
442
}
443