From bf90f5982cde20ca10b7b0f5f4256ff8965781bd Mon Sep 17 00:00:00 2001 From: npldevfr Date: Mon, 22 Jan 2024 11:21:24 +0100 Subject: [PATCH] feat: Add Endpoints and Query classes --- composer.json | 4 +- src/Endpoints/Endpoints.php | 23 +++ src/Query.php | 221 ++++++++++++++++++++++++++++ src/Query/QueryBuilder.php | 21 +++ src/Query/QueryBuilderInterface.php | 17 +++ tests/QueryBuilderTest.php | 31 ++++ 6 files changed, 315 insertions(+), 2 deletions(-) create mode 100644 src/Endpoints/Endpoints.php create mode 100644 src/Query.php create mode 100644 src/Query/QueryBuilder.php create mode 100644 src/Query/QueryBuilderInterface.php create mode 100644 tests/QueryBuilderTest.php diff --git a/composer.json b/composer.json index 405c79c..02b238e 100644 --- a/composer.json +++ b/composer.json @@ -10,13 +10,13 @@ } ], "require": { - "php": "^8.2.0" + "php": "^8.2.0", + "guzzlehttp/guzzle": "^7.0" }, "require-dev": { "laravel/pint": "^1.13.7", "pestphp/pest": "^2.28.1", "phpstan/phpstan": "^1.10.50", - "rector/rector": "^0.19.2", "symfony/var-dumper": "^6.4.0|^7.0.0" }, "autoload": { diff --git a/src/Endpoints/Endpoints.php b/src/Endpoints/Endpoints.php new file mode 100644 index 0000000..bd29724 --- /dev/null +++ b/src/Endpoints/Endpoints.php @@ -0,0 +1,23 @@ + $params + * @param Client|null $client + */ + public function __construct(array $params = [], ?Client $client = null) + { + parent::__construct(array_merge([ + 'wiki' => '', + 'limit' => 100, + 'conditions' => '', + ], $params)); + + $this->client = $client ?? new Client([ + 'base_uri' => 'https://api.liquipedia.net/api/', + ]); + } + + /** + * @param array $wikis + */ + public function wikis(array $wikis): self + { + + $wikis = implode('|', $wikis); + $wikis = preg_replace('/\|+/', '|', $wikis); + + $this->params['wiki'] = $wikis; + + return $this; + } + + public function addWiki(string $wiki): self + { + $this->params['wiki'] .= '|'.$wiki; + + return $this; + } + + public function removeWiki(string $wiki): self + { + $this->params['wiki'] = preg_replace( + '/\|+/', + '|', + str_replace( + $wiki, + '', + $this->params['wiki'] + ) + ); + + return $this; + } + + public function limit(int $limit): self + { + $this->params['limit'] = $limit; + + return $this; + } + + public function rawConditions(string $conditions): self + { + $this->params['conditions'] = trim($this->params['conditions'].' '.$conditions); + + return $this; + } + + /** + * @throws Exception + */ + public function andCondition(string $key, string $operator, string $value): self + { + // operator are : :: (equals), ::! (not equals), ::< (lower than) or ::> (greater than). + if (! in_array($operator, ['::', '::!', '::<', '::>'])) { + throw new Exception('Operator must be ::, ::!, ::< or ::>'); + } + + $this->params['conditions'] = trim($this->params['conditions'].' AND '."[[{$key}{$operator}{$value}]]"); + + return $this; + } + + /** + * @param array $values + */ + public function andConditions(string $key, string $operator, array $values): self + { + // operator are : :: (equals), ::! (not equals), ::< (lower than) or ::> (greater than). + if (! in_array($operator, ['::', '::!', '::<', '::>'])) { + throw new Exception('Operator must be ::, ::!, ::< or ::>'); + } + + $conditions = []; + foreach ($values as $value) { + $conditions[] = "[[{$key}{$operator}{$value}]]"; + } + + $this->params['conditions'] = trim($this->params['conditions'].' '.implode(' AND ', $conditions)); + + return $this; + } + + public function orCondition(string $key, string $operator, string $value): self + { + // operator are : :: (equals), ::! (not equals), ::< (lower than) or ::> (greater than). + if (! in_array($operator, ['::', '::!', '::<', '::>'])) { + throw new Exception('Operator must be ::, ::!, ::< or ::>'); + } + + $this->params['conditions'] = trim($this->params['conditions'].' OR '."[[{$key}{$operator}{$value}]]"); + + return $this; + } + + public function orConditions(string $key, string $operator, array $values): self + { + // operator are : :: (equals), ::! (not equals), ::< (lower than) or ::> (greater than). + if (! in_array($operator, ['::', '::!', '::<', '::>'])) { + throw new Exception('Operator must be ::, ::!, ::< or ::>'); + } + + $conditions = []; + foreach ($values as $value) { + $conditions[] = "[[{$key}{$operator}{$value}]]"; + } + + $this->params['conditions'] = trim($this->params['conditions'].' OR '.implode(' OR ', $conditions)); + $this->params['conditions'] = trim(preg_replace('/^OR/', '', $this->params['conditions'])); + + return $this; + } + + public function select(array $fields): self + { + if (! isset($this->params['query'])) { + $this->params['query'] = implode(',', $fields); + } else { + $this->params['query'] .= ','.implode(',', $fields); + } + + return $this; + } + + /** + * What you want your results grouped by (this can be helpful when using aggregate functions). + * Example: pagename ASC + * + * @throws Exception + */ + public function groupBy(string $field, string $direction = 'ASC'): self + { + + if (! in_array(strtoupper($direction), ['ASC', 'DESC'])) { + throw new Exception('Direction must be ASC or DESC'); + } + + $this->params['groupby'] = "{$field} {$direction}"; + + return $this; + } + + /** + * The order you want your result in. + * Example: pagename ASC + * + * @throws Exception + */ + public function orderBy(string $field, string $direction = 'ASC'): self + { + + if (! in_array(strtoupper($direction), ['ASC', 'DESC'])) { + throw new Exception('Direction must be ASC or DESC'); + } + + $this->params['order'] = "{$field} {$direction}"; + + return $this; + } + + public function setEndpoint(string $endpoint): self + { + $this->endpoint = $endpoint; + + return $this; + } + + /** + * @throws GuzzleException + * @throws Exception + */ + public function get(?string $endpoint = null): array + { + + $customEndpoint = $endpoint ?? $this->endpoint; + + $response = json_decode($this->client->get($customEndpoint, [ + 'query' => $this->params, + ])->getBody()->getContents(), false, 512, JSON_THROW_ON_ERROR); + + if (isset($response->error)) { + throw new Exception($response->error); + } + + return $response->result ?? []; + } +} diff --git a/src/Query/QueryBuilder.php b/src/Query/QueryBuilder.php new file mode 100644 index 0000000..48ccdae --- /dev/null +++ b/src/Query/QueryBuilder.php @@ -0,0 +1,21 @@ + + */ + protected array $params; + + public function __construct(array $params = []) + { + $this->params = $params; + } + + public function build(): array + { + return $this->params; + } +} diff --git a/src/Query/QueryBuilderInterface.php b/src/Query/QueryBuilderInterface.php new file mode 100644 index 0000000..efff368 --- /dev/null +++ b/src/Query/QueryBuilderInterface.php @@ -0,0 +1,17 @@ + $params + */ + public function __construct(array $params = []); + + /** + * @return array + */ + public function build(): array; +} diff --git a/tests/QueryBuilderTest.php b/tests/QueryBuilderTest.php new file mode 100644 index 0000000..c65b8b3 --- /dev/null +++ b/tests/QueryBuilderTest.php @@ -0,0 +1,31 @@ +toBeInstanceOf(QueryBuilder::class); +}); + +it('can be instantiated with params', function () { + $queryBuilder = new class(['foo' => 'bar']) extends QueryBuilder { + }; + + expect($queryBuilder->build())->toBe(['foo' => 'bar']); +}); + +it('can be instantiated with params and build', function () { + $queryBuilder = new class(['foo' => 'bar']) extends QueryBuilder { + public function build(): array + { + return ['bar' => 'foo']; + } + }; + + expect($queryBuilder->build())->toBe(['bar' => 'foo']); +}); + +