Add Flux from() and filter() methods
Signed-off-by: davidarendsen <davidarendsen@hey.com>
This commit is contained in:
parent
af750eb635
commit
e0e69b4beb
17 changed files with 2095 additions and 0 deletions
1
.gitignore
vendored
1
.gitignore
vendored
|
|
@ -1,5 +1,6 @@
|
|||
composer.phar
|
||||
/vendor/
|
||||
.phpunit.result.cache
|
||||
|
||||
# Commit your application's lock file https://getcomposer.org/doc/01-basic-usage.md#commit-your-composer-lock-file-to-version-control
|
||||
# You may choose to ignore a library lock file http://getcomposer.org/doc/02-libraries.md#lock-file
|
||||
|
|
|
|||
26
composer.json
Normal file
26
composer.json
Normal file
|
|
@ -0,0 +1,26 @@
|
|||
{
|
||||
"name": "arendsen/fluxquerybuilder",
|
||||
"description": "Generating Flux queries",
|
||||
"keywords": [
|
||||
"influxdb",
|
||||
"flux"
|
||||
],
|
||||
"license": "MIT",
|
||||
"authors": [
|
||||
{
|
||||
"name": "David Arendsen",
|
||||
"email": "davidarendsen@hey.com"
|
||||
}
|
||||
],
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"Arendsen\\FluxQueryBuilder\\": "src/"
|
||||
}
|
||||
},
|
||||
"require": {
|
||||
"php": ">=7.2"
|
||||
},
|
||||
"require-dev": {
|
||||
"phpunit/phpunit": "^8"
|
||||
}
|
||||
}
|
||||
1716
composer.lock
generated
Normal file
1716
composer.lock
generated
Normal file
File diff suppressed because it is too large
Load diff
10
phpunit.xml
Normal file
10
phpunit.xml
Normal file
|
|
@ -0,0 +1,10 @@
|
|||
<?xml version="1.0" encoding="utf-8" ?>
|
||||
<phpunit bootstrap="./vendor/autoload.php">
|
||||
|
||||
<testsuites>
|
||||
<testsuite name="The project's test suite">
|
||||
<directory>./tests</directory>
|
||||
</testsuite>
|
||||
</testsuites>
|
||||
|
||||
</phpunit>
|
||||
15
src/Expression/Base.php
Normal file
15
src/Expression/Base.php
Normal file
|
|
@ -0,0 +1,15 @@
|
|||
<?php
|
||||
|
||||
namespace Arendsen\FluxQueryBuilder\Expression;
|
||||
|
||||
abstract class Base {
|
||||
|
||||
/**
|
||||
* @throws ExpressionNotImplementedException
|
||||
*/
|
||||
public function __toString()
|
||||
{
|
||||
throw new ExpressionNotImplementedException('__toString', get_class($this));
|
||||
}
|
||||
|
||||
}
|
||||
7
src/Expression/ExpressionNotImplementedException.php
Normal file
7
src/Expression/ExpressionNotImplementedException.php
Normal file
|
|
@ -0,0 +1,7 @@
|
|||
<?php
|
||||
|
||||
namespace Arendsen\FluxQueryBuilder\Expression;
|
||||
|
||||
use Exception;
|
||||
|
||||
class ExpressionNotImplementedException extends Exception {}
|
||||
30
src/Expression/Filter/AndExpression.php
Normal file
30
src/Expression/Filter/AndExpression.php
Normal file
|
|
@ -0,0 +1,30 @@
|
|||
<?php
|
||||
|
||||
namespace Arendsen\FluxQueryBuilder\Expression\Filter;
|
||||
|
||||
use Arendsen\FluxQueryBuilder\Expression\Base;
|
||||
|
||||
class AndExpression extends Base {
|
||||
|
||||
/**
|
||||
* @var string $key
|
||||
*/
|
||||
private $key;
|
||||
|
||||
/**
|
||||
* @var string $value
|
||||
*/
|
||||
private $value;
|
||||
|
||||
public function __construct(string $key, string $value)
|
||||
{
|
||||
$this->key = $key;
|
||||
$this->value = $value;
|
||||
}
|
||||
|
||||
public function __toString()
|
||||
{
|
||||
return 'and r.' . $this->key . ' == "' . $this->value . '"';
|
||||
}
|
||||
|
||||
}
|
||||
24
src/Expression/Filter/Measurement.php
Normal file
24
src/Expression/Filter/Measurement.php
Normal file
|
|
@ -0,0 +1,24 @@
|
|||
<?php
|
||||
|
||||
namespace Arendsen\FluxQueryBuilder\Expression\Filter;
|
||||
|
||||
use Arendsen\FluxQueryBuilder\Expression\Base;
|
||||
|
||||
class Measurement extends Base {
|
||||
|
||||
/**
|
||||
* @var array $measurement
|
||||
*/
|
||||
private $measurement;
|
||||
|
||||
public function __construct(string $measurement)
|
||||
{
|
||||
$this->measurement = $measurement;
|
||||
}
|
||||
|
||||
public function __toString()
|
||||
{
|
||||
return 'r._measurement == "' . $this->measurement . '"';
|
||||
}
|
||||
|
||||
}
|
||||
30
src/Expression/Filter/OrExpression.php
Normal file
30
src/Expression/Filter/OrExpression.php
Normal file
|
|
@ -0,0 +1,30 @@
|
|||
<?php
|
||||
|
||||
namespace Arendsen\FluxQueryBuilder\Expression\Filter;
|
||||
|
||||
use Arendsen\FluxQueryBuilder\Expression\Base;
|
||||
|
||||
class OrExpression extends Base {
|
||||
|
||||
/**
|
||||
* @var string $key
|
||||
*/
|
||||
private $key;
|
||||
|
||||
/**
|
||||
* @var string $value
|
||||
*/
|
||||
private $value;
|
||||
|
||||
public function __construct(string $key, string $value)
|
||||
{
|
||||
$this->key = $key;
|
||||
$this->value = $value;
|
||||
}
|
||||
|
||||
public function __toString()
|
||||
{
|
||||
return 'or r.' . $this->key . ' == "' . $this->value . '"';
|
||||
}
|
||||
|
||||
}
|
||||
15
src/Function/Base.php
Normal file
15
src/Function/Base.php
Normal file
|
|
@ -0,0 +1,15 @@
|
|||
<?php
|
||||
|
||||
namespace Arendsen\FluxQueryBuilder\Function;
|
||||
|
||||
abstract class Base {
|
||||
|
||||
/**
|
||||
* @throws FunctionNotImplementedException
|
||||
*/
|
||||
public function __toString()
|
||||
{
|
||||
throw new FunctionNotImplementedException('__toString', get_class($this));
|
||||
}
|
||||
|
||||
}
|
||||
22
src/Function/Filter.php
Normal file
22
src/Function/Filter.php
Normal file
|
|
@ -0,0 +1,22 @@
|
|||
<?php
|
||||
|
||||
namespace Arendsen\FluxQueryBuilder\Function;
|
||||
|
||||
class Filter extends Base {
|
||||
|
||||
/**
|
||||
* @var array $settings
|
||||
*/
|
||||
private $settings;
|
||||
|
||||
public function __construct(array $settings)
|
||||
{
|
||||
$this->settings = $settings;
|
||||
}
|
||||
|
||||
public function __toString()
|
||||
{
|
||||
return '|> filter(fn: (r) => ' . implode(' ', $this->settings) . ') ';
|
||||
}
|
||||
|
||||
}
|
||||
31
src/Function/From.php
Normal file
31
src/Function/From.php
Normal file
|
|
@ -0,0 +1,31 @@
|
|||
<?php
|
||||
|
||||
namespace Arendsen\FluxQueryBuilder\Function;
|
||||
|
||||
class From extends Base {
|
||||
|
||||
/**
|
||||
* @var array $settings
|
||||
*/
|
||||
private $settings;
|
||||
|
||||
public function __construct(array $settings)
|
||||
{
|
||||
$this->settings = $settings;
|
||||
}
|
||||
|
||||
public function __toString()
|
||||
{
|
||||
return 'from(' . implode(', ', $this->format($this->settings)) . ') ';
|
||||
}
|
||||
|
||||
protected function format(array $settings)
|
||||
{
|
||||
array_walk($settings, function(&$value, $key) {
|
||||
$value = $key . ': ' . (is_string($value) ? '"' . $value . '"' : $value);
|
||||
});
|
||||
|
||||
return $settings;
|
||||
}
|
||||
|
||||
}
|
||||
7
src/Function/FunctionNotImplementedException.php
Normal file
7
src/Function/FunctionNotImplementedException.php
Normal file
|
|
@ -0,0 +1,7 @@
|
|||
<?php
|
||||
|
||||
namespace Arendsen\FluxQueryBuilder\Function;
|
||||
|
||||
use Exception;
|
||||
|
||||
class FunctionNotImplementedException extends Exception {}
|
||||
57
src/QueryBuilder.php
Normal file
57
src/QueryBuilder.php
Normal file
|
|
@ -0,0 +1,57 @@
|
|||
<?php
|
||||
|
||||
namespace Arendsen\FluxQueryBuilder;
|
||||
|
||||
class QueryBuilder {
|
||||
|
||||
/**
|
||||
* @var array $from
|
||||
*/
|
||||
private $from;
|
||||
|
||||
/**
|
||||
* @var string $measurement
|
||||
*/
|
||||
private $measurement;
|
||||
|
||||
/**
|
||||
* @var array $range
|
||||
*/
|
||||
private $range;
|
||||
|
||||
public function from(array $from): QueryBuilder
|
||||
{
|
||||
$this->from = $from;
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function fromBucket(string $bucket): QueryBuilder {
|
||||
$this->from = ['bucket' => $bucket];
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function fromMeasurement(string $measurement): QueryBuilder
|
||||
{
|
||||
$this->measurement = $measurement;
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function addRange(array $range): QueryBuilder
|
||||
{
|
||||
$this->range = $range;
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function addRangeStart(string $rangeStart): QueryBuilder
|
||||
{
|
||||
$this->range = ['start' => $rangeStart];
|
||||
return $this;
|
||||
}
|
||||
|
||||
public function build(): string
|
||||
{
|
||||
//TODO: build the query dynamically here
|
||||
return 'from(bucket: "test_bucket", host: "host", org: "example-org", token: "token") |> ';
|
||||
}
|
||||
|
||||
}
|
||||
27
tests/Expression/FilterExpressionTest.php
Normal file
27
tests/Expression/FilterExpressionTest.php
Normal file
|
|
@ -0,0 +1,27 @@
|
|||
<?php
|
||||
declare(strict_types=1);
|
||||
|
||||
use Arendsen\FluxQueryBuilder\Expression\Filter\AndExpression;
|
||||
use Arendsen\FluxQueryBuilder\Expression\Filter\OrExpression;
|
||||
use Arendsen\FluxQueryBuilder\Expression\Filter\Measurement;
|
||||
use Arendsen\FluxQueryBuilder\Function\Filter;
|
||||
use PHPUnit\Framework\TestCase;
|
||||
|
||||
final class FilterFunctionTest extends TestCase {
|
||||
|
||||
public function testSimpleFilter()
|
||||
{
|
||||
$expression = new Filter([
|
||||
new Measurement('test_measurement'),
|
||||
new AndExpression('_field', 'user'),
|
||||
new OrExpression('_field', 'field2'),
|
||||
'and r.user == "my_username"',
|
||||
]);
|
||||
|
||||
$query = '|> filter(fn: (r) => r._measurement == "test_measurement" and r._field == "user" or ' .
|
||||
'r._field == "field2" and r.user == "my_username") ';
|
||||
|
||||
$this->assertEquals($expression->__toString(), $query);
|
||||
}
|
||||
|
||||
}
|
||||
38
tests/Expression/FromExpressionTest.php
Normal file
38
tests/Expression/FromExpressionTest.php
Normal file
|
|
@ -0,0 +1,38 @@
|
|||
<?php
|
||||
declare(strict_types=1);
|
||||
|
||||
use Arendsen\FluxQueryBuilder\Function\From;
|
||||
use PHPUnit\Framework\TestCase;
|
||||
|
||||
final class FromExpressionTest extends TestCase {
|
||||
|
||||
/**
|
||||
* @dataProvider somethingProvider
|
||||
*/
|
||||
public function testSomething($settings, $query)
|
||||
{
|
||||
$expression = new From($settings);
|
||||
|
||||
$this->assertEquals($expression->__toString(), $query);
|
||||
}
|
||||
|
||||
public function somethingProvider(): array
|
||||
{
|
||||
return [
|
||||
'from bucket' => [
|
||||
[
|
||||
'bucket' => 'test-bucket',
|
||||
],
|
||||
'from(bucket: "test-bucket") '
|
||||
],
|
||||
'from bucket and host' => [
|
||||
[
|
||||
'bucket' => 'test-bucket',
|
||||
'host' => 'test',
|
||||
],
|
||||
'from(bucket: "test-bucket", host: "test") '
|
||||
],
|
||||
];
|
||||
}
|
||||
|
||||
}
|
||||
39
tests/QueryBuilderTest.php
Normal file
39
tests/QueryBuilderTest.php
Normal file
|
|
@ -0,0 +1,39 @@
|
|||
<?php
|
||||
declare(strict_types=1);
|
||||
|
||||
use PHPUnit\Framework\TestCase;
|
||||
use Arendsen\FluxQueryBuilder\QueryBuilder;
|
||||
|
||||
final class QueryBuilderTest extends TestCase {
|
||||
|
||||
/**
|
||||
* @dataProvider somethingProvider
|
||||
*/
|
||||
public function testSomething($bucket, $measurement, $range, $expectedQuery)
|
||||
{
|
||||
$queryBuilder = new QueryBuilder();
|
||||
$queryBuilder->from($bucket)
|
||||
->fromMeasurement($measurement)
|
||||
->addRangeStart($range);
|
||||
|
||||
$this->assertEquals($queryBuilder->build(), $expectedQuery);
|
||||
}
|
||||
|
||||
public function somethingProvider(): array
|
||||
{
|
||||
return [
|
||||
'case 1' => [
|
||||
[
|
||||
'bucket' => 'example-bucket',
|
||||
'host' => 'host',
|
||||
'org' => 'example-org',
|
||||
'token' => 'token'
|
||||
],
|
||||
'test_measurement',
|
||||
'-360h',
|
||||
'from(bucket: "test_bucket", host: "host", org: "example-org", token: "token") |> '
|
||||
],
|
||||
];
|
||||
}
|
||||
|
||||
}
|
||||
Loading…
Add table
editor.link_modal.header
Reference in a new issue