Skip to content

Commit fa6782c

Browse files
Merge pull request #831 from spatie/auto-lazy
Auto lazy
2 parents debae6f + 46f9551 commit fa6782c

13 files changed

+568
-23
lines changed

docs/advanced-usage/use-with-inertia.md

+35
Original file line numberDiff line numberDiff line change
@@ -52,3 +52,38 @@ router.reload((url, {
5252
only: ['title'],
5353
});
5454
```
55+
56+
### Auto lazy Inertia properties
57+
58+
We already saw earlier that the package can automatically make properties Lazy, the same can be done for Inertia properties.
59+
60+
It is possible to rewrite the previous example as follows:
61+
62+
```php
63+
use Spatie\LaravelData\Attributes\AutoClosureLazy;use Spatie\LaravelData\Attributes\AutoInertiaLazy;
64+
65+
class SongData extends Data
66+
{
67+
public function __construct(
68+
#[AutoInertiaLazy]
69+
public Lazy|string $title,
70+
#[AutoClosureLazy]
71+
public Lazy|string $artist,
72+
) {
73+
}
74+
}
75+
```
76+
77+
If all the properties of a class should be either Inertia or closure lazy, you can use the attributes on the class level:
78+
79+
```php
80+
#[AutoInertiaLazy]
81+
class SongData extends Data
82+
{
83+
public function __construct(
84+
public Lazy|string $title,
85+
public Lazy|string $artist,
86+
) {
87+
}
88+
}
89+
```

docs/as-a-resource/lazy-properties.md

+144-13
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,8 @@ class AlbumData extends Data
1919
}
2020
```
2121

22-
This will always output a collection of songs, which can become quite large. With lazy properties, we can include properties when we want to:
22+
This will always output a collection of songs, which can become quite large. With lazy properties, we can include
23+
properties when we want to:
2324

2425
```php
2526
class AlbumData extends Data
@@ -43,7 +44,8 @@ class AlbumData extends Data
4344
}
4445
```
4546

46-
The `songs` key won't be included in the resource when transforming it from a model. Because the closure that provides the data won't be called when transforming the data object unless we explicitly demand it.
47+
The `songs` key won't be included in the resource when transforming it from a model. Because the closure that provides
48+
the data won't be called when transforming the data object unless we explicitly demand it.
4749

4850
Now when we transform the data object as such:
4951

@@ -69,7 +71,8 @@ AlbumData::from(Album::first())->include('songs');
6971

7072
Lazy properties will only be included when the `include` method is called on the data object with the property's name.
7173

72-
It is also possible to nest these includes. For example, let's update the `SongData` class and make all of its properties lazy:
74+
It is also possible to nest these includes. For example, let's update the `SongData` class and make all of its
75+
properties lazy:
7376

7477
```php
7578
class SongData extends Data
@@ -108,7 +111,8 @@ If you want to include all the properties of a data object, you can do the follo
108111
AlbumData::from(Album::first())->include('songs.*');
109112
```
110113

111-
Explicitly including properties of data objects also works on a single data object. For example, our `UserData` looks like this:
114+
Explicitly including properties of data objects also works on a single data object. For example, our `UserData` looks
115+
like this:
112116

113117
```php
114118
class UserData extends Data
@@ -147,13 +151,15 @@ Lazy::create(fn() => SongData::collect($album->songs));
147151

148152
With a basic `Lazy` property, you must explicitly include it when the data object is transformed.
149153

150-
Sometimes you only want to include a property when a specific condition is true. This can be done with conditional lazy properties:
154+
Sometimes you only want to include a property when a specific condition is true. This can be done with conditional lazy
155+
properties:
151156

152157
```php
153158
Lazy::when(fn() => $this->is_admin, fn() => SongData::collect($album->songs));
154159
```
155160

156-
The property will only be included when the `is_admin` property of the data object is true. It is not possible to include the property later on with the `include` method when a condition is not accepted.
161+
The property will only be included when the `is_admin` property of the data object is true. It is not possible to
162+
include the property later on with the `include` method when a condition is not accepted.
157163

158164
### Relational Lazy properties
159165

@@ -173,15 +179,138 @@ It is possible to mark a lazy property as included by default:
173179
Lazy::create(fn() => SongData::collect($album->songs))->defaultIncluded();
174180
```
175181

176-
The property will now always be included when the data object is transformed. You can explicitly exclude properties that were default included as such:
182+
The property will now always be included when the data object is transformed. You can explicitly exclude properties that
183+
were default included as such:
177184

178185
```php
179186
AlbumData::create(Album::first())->exclude('songs');
180187
```
181188

189+
## Auto Lazy
190+
191+
Writing Lazy properties can be a bit cumbersome. It is often a repetitive task to write the same code over and over
192+
again while the package can infer almost everything.
193+
194+
Let's take a look at our previous example:
195+
196+
```php
197+
class UserData extends Data
198+
{
199+
public function __construct(
200+
public string $title,
201+
public Lazy|SongData $favorite_song,
202+
) {
203+
}
204+
205+
public static function fromModel(User $user): self
206+
{
207+
return new self(
208+
$user->title,
209+
Lazy::create(fn() => SongData::from($user->favorite_song))
210+
);
211+
}
212+
}
213+
```
214+
215+
The package knows how to get the property from the model and wrap it into a data object, but since we're using a lazy
216+
property, we need to write our own magic creation method with a lot of repetitive code.
217+
218+
In such a situation auto lazy might be a good fit, instead of casting the property directly into the data object, the
219+
casting process is wrapped in a lazy Closure.
220+
221+
This makes it possible to rewrite the example as such:
222+
223+
```php
224+
#[AutoLazy]
225+
class UserData extends Data
226+
{
227+
public function __construct(
228+
public string $title,
229+
public Lazy|SongData $favorite_song,
230+
) {
231+
}
232+
}
233+
```
234+
235+
While achieving the same result!
236+
237+
Auto Lazy wraps the casting process of a value for every property typed as `Lazy` into a Lazy Closure when the
238+
`AutoLazy` attribute is present on the class.
239+
240+
It is also possible to use the `AutoLazy` attribute on a property level:
241+
242+
```php
243+
class UserData extends Data
244+
{
245+
public function __construct(
246+
public string $title,
247+
#[AutoLazy]
248+
public Lazy|SongData $favorite_song,
249+
) {
250+
}
251+
}
252+
```
253+
254+
The auto lazy process won't be applied in the following situations:
255+
256+
- When a null value is passed to the property
257+
- When the property value isn't present in the input payload and the property typed as `Optional`
258+
- When a Lazy Closure is passed to the property
259+
260+
### Auto lazy with model relations
261+
262+
When you're constructing a data object from an Eloquent model, it is also possible to automatically create lazy
263+
properties for model relations which are only resolved when the relation is loaded:
264+
265+
```php
266+
class UserData extends Data
267+
{
268+
public function __construct(
269+
public string $title,
270+
#[AutoWhenLoadedLazy]
271+
public Lazy|SongData $favoriteSong,
272+
) {
273+
}
274+
}
275+
```
276+
277+
When the `favoriteSong` relation is loaded on the model, the property will be included in the data object.
278+
279+
If the name of the relation doesn't match the property name, you can specify the relation name:
280+
281+
```php
282+
class UserData extends Data
283+
{
284+
public function __construct(
285+
public string $title,
286+
#[AutoWhenLoadedLazy('favoriteSong')]
287+
public Lazy|SongData $favorite_song,
288+
) {
289+
}
290+
}
291+
```
292+
293+
The package will use the regular casting process when the relation is loaded, so it is also perfectly possible to create a collection of data objects:
294+
295+
```php
296+
class UserData extends Data
297+
{
298+
/**
299+
* @param Lazy|array<int, SongData> $favoriteSongs
300+
*/
301+
public function __construct(
302+
public string $title,
303+
#[AutoWhenLoadedLazy]
304+
public Lazy|array $favoriteSongs,
305+
) {
306+
}
307+
}
308+
```
309+
182310
## Only and Except
183311

184-
Lazy properties are great for reducing payloads sent over the wire. However, when you completely want to remove a property Laravel's `only` and `except` methods can be used:
312+
Lazy properties are great for reducing payloads sent over the wire. However, when you completely want to remove a
313+
property Laravel's `only` and `except` methods can be used:
185314

186315
```php
187316
AlbumData::from(Album::first())->only('songs'); // will only show `songs`
@@ -202,7 +331,8 @@ AlbumData::from(Album::first())->only('songs.{name, artist}');
202331
AlbumData::from(Album::first())->except('songs.{name, artist}');
203332
```
204333

205-
Only and except always take precedence over include and exclude, which means that when a property is hidden by `only` or `except` it is impossible to show it again using `include`.
334+
Only and except always take precedence over include and exclude, which means that when a property is hidden by `only` or
335+
`except` it is impossible to show it again using `include`.
206336

207337
### Conditionally
208338

@@ -306,7 +436,7 @@ Our JSON would look like this when we request `https://spatie.be/my-account`:
306436

307437
```json
308438
{
309-
"name": "Ruben Van Assche"
439+
"name" : "Ruben Van Assche"
310440
}
311441
```
312442

@@ -318,8 +448,8 @@ https://spatie.be/my-account?include=favorite_song
318448

319449
```json
320450
{
321-
"name": "Ruben Van Assche",
322-
"favorite_song": {
451+
"name" : "Ruben Van Assche",
452+
"favorite_song" : {
323453
"name" : "Never Gonna Give You Up",
324454
"artist" : "Rick Astley"
325455
}
@@ -395,7 +525,8 @@ AlbumData::from(Album::first())->include('songs')->toArray(); // will include so
395525
AlbumData::from(Album::first())->toArray(); // will not include songs
396526
```
397527

398-
If you want to add includes/excludes/only/except to a data object and its nested chain that will be used for all future transformations, you can define them in their respective *properties methods:
528+
If you want to add includes/excludes/only/except to a data object and its nested chain that will be used for all future
529+
transformations, you can define them in their respective *properties methods:
399530

400531
```php
401532
class AlbumData extends Data

src/Attributes/AutoClosureLazy.php

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
<?php
2+
3+
namespace Spatie\LaravelData\Attributes;
4+
5+
use Attribute;
6+
use Closure;
7+
use Spatie\LaravelData\Lazy;
8+
use Spatie\LaravelData\Support\DataProperty;
9+
use Spatie\LaravelData\Support\Lazy\ClosureLazy;
10+
11+
#[Attribute(Attribute::TARGET_CLASS | Attribute::TARGET_PROPERTY)]
12+
class AutoClosureLazy extends AutoLazy
13+
{
14+
public function build(Closure $castValue, mixed $payload, DataProperty $property, mixed $value): ClosureLazy
15+
{
16+
return Lazy::closure(fn () => $castValue($value));
17+
}
18+
}

src/Attributes/AutoInertiaLazy.php

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
<?php
2+
3+
namespace Spatie\LaravelData\Attributes;
4+
5+
use Attribute;
6+
use Closure;
7+
use Spatie\LaravelData\Lazy;
8+
use Spatie\LaravelData\Support\DataProperty;
9+
use Spatie\LaravelData\Support\Lazy\InertiaLazy;
10+
11+
#[Attribute(Attribute::TARGET_CLASS | Attribute::TARGET_PROPERTY)]
12+
class AutoInertiaLazy extends AutoLazy
13+
{
14+
public function build(Closure $castValue, mixed $payload, DataProperty $property, mixed $value): InertiaLazy
15+
{
16+
return Lazy::inertia(fn () => $castValue($value));
17+
}
18+
}

src/Attributes/AutoLazy.php

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
<?php
2+
3+
namespace Spatie\LaravelData\Attributes;
4+
5+
use Attribute;
6+
use Closure;
7+
use Spatie\LaravelData\Lazy;
8+
use Spatie\LaravelData\Support\DataProperty;
9+
10+
#[Attribute(Attribute::TARGET_CLASS | Attribute::TARGET_PROPERTY)]
11+
class AutoLazy
12+
{
13+
public function build(Closure $castValue, mixed $payload, DataProperty $property, mixed $value): Lazy
14+
{
15+
return Lazy::create(fn () => $castValue($value));
16+
}
17+
}

src/Attributes/AutoWhenLoadedLazy.php

+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
<?php
2+
3+
namespace Spatie\LaravelData\Attributes;
4+
5+
use Attribute;
6+
use Closure;
7+
use Spatie\LaravelData\Lazy;
8+
use Spatie\LaravelData\Support\DataProperty;
9+
use Spatie\LaravelData\Support\Lazy\ConditionalLazy;
10+
11+
#[Attribute(Attribute::TARGET_PROPERTY)]
12+
class AutoWhenLoadedLazy extends AutoLazy
13+
{
14+
public function __construct(
15+
public ?string $relation = null,
16+
) {
17+
}
18+
19+
public function build(Closure $castValue, mixed $payload, DataProperty $property, mixed $value): ConditionalLazy
20+
{
21+
$relation = $this->relation ?? $property->name;
22+
23+
return Lazy::when(fn () => $payload->relationLoaded($relation), fn () => $castValue(
24+
$payload->getRelation($relation)
25+
));
26+
}
27+
}

src/DataPipes/CastPropertiesDataPipe.php

+22-1
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,28 @@ public function handle(
4040
continue;
4141
}
4242

43-
$properties[$name] = $this->cast($dataProperty, $value, $properties, $creationContext);
43+
if ($dataProperty->autoLazy) {
44+
$properties[$name] = $dataProperty->autoLazy->build(
45+
fn (mixed $value) => $this->cast(
46+
$dataProperty,
47+
$value,
48+
$properties,
49+
$creationContext
50+
),
51+
$payload,
52+
$dataProperty,
53+
$value
54+
);
55+
56+
continue;
57+
}
58+
59+
$properties[$name] = $this->cast(
60+
$dataProperty,
61+
$value,
62+
$properties,
63+
$creationContext
64+
);
4465
}
4566

4667
return $properties;

0 commit comments

Comments
 (0)