Skip to content

Commit 35ccb63

Browse files
committed
Merge branch 'release/4.2.0'
2 parents c4f86c7 + a30eb76 commit 35ccb63

11 files changed

+2333
-2
lines changed

CHANGELOG.md

+6
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,12 @@ All notable changes to this project will be documented in this file. This projec
55

66
## Unreleased
77

8+
## [4.2.0] - 2024-08-26
9+
10+
### Added
11+
12+
- [#37](https://github.com/laravel-json-api/eloquent/pull/37) Add Eloquent cursor pagination implementation.
13+
814
## [4.1.0] - 2024-06-26
915

1016
### Added

src/Pagination/Cursor/Cursor.php

+93
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,93 @@
1+
<?php
2+
/*
3+
* Copyright 2024 Cloud Creativity Limited
4+
*
5+
* Use of this source code is governed by an MIT-style
6+
* license that can be found in the LICENSE file or at
7+
* https://opensource.org/licenses/MIT.
8+
*/
9+
10+
declare(strict_types=1);
11+
12+
namespace LaravelJsonApi\Eloquent\Pagination\Cursor;
13+
14+
use InvalidArgumentException;
15+
16+
final readonly class Cursor
17+
{
18+
/**
19+
* Cursor constructor.
20+
*
21+
* @param string|null $before
22+
* @param string|null $after
23+
* @param int|null $limit
24+
*/
25+
public function __construct(
26+
private ?string $before = null,
27+
private ?string $after = null,
28+
private ?int $limit = null
29+
) {
30+
if (is_int($this->limit) && 1 > $this->limit) {
31+
throw new InvalidArgumentException('Expecting a limit that is 1 or greater.');
32+
}
33+
}
34+
35+
/**
36+
* @return bool
37+
*/
38+
public function isBefore(): bool
39+
{
40+
return !is_null($this->before);
41+
}
42+
43+
/**
44+
* @return string|null
45+
*/
46+
public function getBefore(): ?string
47+
{
48+
return $this->before;
49+
}
50+
51+
/**
52+
* @return bool
53+
*/
54+
public function isAfter(): bool
55+
{
56+
return !is_null($this->after) && !$this->isBefore();
57+
}
58+
59+
/**
60+
* @return string|null
61+
*/
62+
public function getAfter(): ?string
63+
{
64+
return $this->after;
65+
}
66+
67+
/**
68+
* Set a limit, if no limit is set on the cursor.
69+
*
70+
* @param int $limit
71+
* @return Cursor
72+
*/
73+
public function withDefaultLimit(int $limit): self
74+
{
75+
if ($this->limit === null) {
76+
return new self(
77+
before: $this->before,
78+
after: $this->after,
79+
limit: $limit,
80+
);
81+
}
82+
83+
return $this;
84+
}
85+
86+
/**
87+
* @return int|null
88+
*/
89+
public function getLimit(): ?int
90+
{
91+
return $this->limit;
92+
}
93+
}
+178
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,178 @@
1+
<?php
2+
/*
3+
* Copyright 2024 Cloud Creativity Limited
4+
*
5+
* Use of this source code is governed by an MIT-style
6+
* license that can be found in the LICENSE file or at
7+
* https://opensource.org/licenses/MIT.
8+
*/
9+
10+
declare(strict_types=1);
11+
12+
namespace LaravelJsonApi\Eloquent\Pagination\Cursor;
13+
14+
use Illuminate\Database\Eloquent\Builder;
15+
use Illuminate\Database\Eloquent\Relations\Relation;
16+
use LaravelJsonApi\Contracts\Schema\ID;
17+
use LaravelJsonApi\Core\Schema\IdParser;
18+
19+
final class CursorBuilder
20+
{
21+
/**
22+
* @var string
23+
*/
24+
private readonly string $keyName;
25+
26+
/**
27+
* @var string
28+
*/
29+
private string $direction;
30+
31+
/**
32+
* @var int|null
33+
*/
34+
private ?int $defaultPerPage = null;
35+
36+
/**
37+
* @var bool
38+
*/
39+
private bool $withTotal;
40+
41+
/**
42+
* @var bool
43+
*/
44+
private bool $keySort = true;
45+
46+
/**
47+
* @var CursorParser
48+
*/
49+
private readonly CursorParser $parser;
50+
51+
/**
52+
* CursorBuilder constructor.
53+
*
54+
* @param Builder|Relation $query the column to use for the cursor
55+
* @param ID $id
56+
* @param string|null $key the key column that the before/after cursors related to
57+
*/
58+
public function __construct(
59+
private readonly Builder|Relation $query,
60+
private readonly ID $id,
61+
?string $key = null
62+
) {
63+
$this->keyName = $key ?: $this->id->key();
64+
$this->parser = new CursorParser(IdParser::make($this->id), $this->keyName);
65+
}
66+
67+
/**
68+
* Set the default number of items per-page.
69+
*
70+
* If null, the default from the `Model::getPage()` method will be used.
71+
*
72+
* @return $this
73+
*/
74+
public function withDefaultPerPage(?int $perPage): self
75+
{
76+
$this->defaultPerPage = $perPage;
77+
78+
return $this;
79+
}
80+
81+
/**
82+
* @param bool $keySort
83+
* @return $this
84+
*/
85+
public function withKeySort(bool $keySort = true): self
86+
{
87+
$this->keySort = $keySort;
88+
89+
return $this;
90+
}
91+
92+
/**
93+
* Set the query direction.
94+
*
95+
* @return $this
96+
*/
97+
public function withDirection(string $direction): self
98+
{
99+
if (\in_array($direction, ['asc', 'desc'])) {
100+
$this->direction = $direction;
101+
102+
return $this;
103+
}
104+
105+
throw new \InvalidArgumentException('Unexpected query direction.');
106+
}
107+
108+
/**
109+
* @param bool $withTotal
110+
* @return $this
111+
*/
112+
public function withTotal(bool $withTotal): self
113+
{
114+
$this->withTotal = $withTotal;
115+
116+
return $this;
117+
}
118+
119+
/**
120+
* @param array<string> $columns
121+
*/
122+
public function paginate(Cursor $cursor, array $columns = ['*']): CursorPaginator
123+
{
124+
$cursor = $cursor->withDefaultLimit($this->getDefaultPerPage());
125+
126+
$this->applyKeySort();
127+
128+
$total = $this->getTotal();
129+
$laravelPaginator = $this->query->cursorPaginate(
130+
$cursor->getLimit(),
131+
$columns,
132+
'cursor',
133+
$this->parser->decode($cursor),
134+
);
135+
$paginator = new CursorPaginator($this->parser, $laravelPaginator, $cursor, $total);
136+
137+
return $paginator->withCurrentPath();
138+
}
139+
140+
/**
141+
* @return void
142+
*/
143+
private function applyKeySort(): void
144+
{
145+
if (!$this->keySort) {
146+
return;
147+
}
148+
149+
if (
150+
empty($this->query->getQuery()->orders)
151+
|| collect($this->query->getQuery()->orders)
152+
->whereIn('column', [$this->keyName, $this->query->qualifyColumn($this->keyName)])
153+
->isEmpty()
154+
) {
155+
$this->query->orderBy($this->keyName, $this->direction);
156+
}
157+
}
158+
159+
/**
160+
* @return int|null
161+
*/
162+
private function getTotal(): ?int
163+
{
164+
return $this->withTotal ? $this->query->count() : null;
165+
}
166+
167+
/**
168+
* @return int
169+
*/
170+
private function getDefaultPerPage(): int
171+
{
172+
if (is_int($this->defaultPerPage)) {
173+
return $this->defaultPerPage;
174+
}
175+
176+
return $this->query->getModel()->getPerPage();
177+
}
178+
}

0 commit comments

Comments
 (0)