Skip to content

Commit

Permalink
Merge pull request #1121 from spatie/fix-enum-casting
Browse files Browse the repository at this point in the history
Fix enum casting
  • Loading branch information
Gummibeer authored Nov 14, 2022
2 parents 10e0f0e + fb36c79 commit eee6143
Show file tree
Hide file tree
Showing 14 changed files with 147 additions and 114 deletions.
2 changes: 1 addition & 1 deletion composer.json
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@
"require": {
"php": "^8.0",
"illuminate/config": "^8.0 || ^9.0",
"illuminate/database": "^8.53 || ^9.0",
"illuminate/database": "^8.69 || ^9.27",
"illuminate/support": "^8.0 || ^9.0",
"spatie/laravel-package-tools": "^1.6.3"
},
Expand Down
39 changes: 0 additions & 39 deletions src/Actions/ResolveForPropertyValueAction.php

This file was deleted.

5 changes: 1 addition & 4 deletions src/ActivityLogger.php
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@
use Illuminate\Support\Str;
use Illuminate\Support\Traits\Conditionable;
use Illuminate\Support\Traits\Macroable;
use Spatie\Activitylog\Actions\ResolveForPropertyValueAction;
use Spatie\Activitylog\Contracts\Activity as ActivityContract;

class ActivityLogger
Expand Down Expand Up @@ -103,15 +102,13 @@ public function setEvent(string $event): static

public function withProperties(mixed $properties): static
{
$this->getActivity()->properties = collect($properties)->map(fn ($value) => ResolveForPropertyValueAction::execute($value));
$this->getActivity()->properties = collect($properties);

return $this;
}

public function withProperty(string $key, mixed $value): static
{
$value = ResolveForPropertyValueAction::execute($value);

$this->getActivity()->properties = $this->getActivity()->properties->put($key, $value);

return $this;
Expand Down
9 changes: 8 additions & 1 deletion src/Traits/LogsActivity.php
Original file line number Diff line number Diff line change
Expand Up @@ -367,7 +367,14 @@ public static function logChanges(Model $model): array
$cast = $model->getCasts()[$attribute];

if (function_exists('enum_exists') && enum_exists($cast)) {
$changes[$attribute] = $model->getStorableEnumValue($changes[$attribute]);
if (method_exists($model, 'getStorableEnumValue')) {
$changes[$attribute] = $model->getStorableEnumValue($changes[$attribute]);
} else {
// ToDo: DEPRECATED - only here for Laravel 8 support
$changes[$attribute] = $changes[$attribute] instanceof \BackedEnum
? $changes[$attribute]->value
: $changes[$attribute]->name;
}
}

if ($model->isCustomDateTimeCast($cast) || $model->isImmutableCustomDateTimeCast($cast)) {
Expand Down
37 changes: 0 additions & 37 deletions tests/AbleStoreNonBackedEnumTest.php

This file was deleted.

24 changes: 18 additions & 6 deletions tests/ActivityLoggerTest.php
Original file line number Diff line number Diff line change
@@ -1,11 +1,13 @@
<?php

use Carbon\Carbon;
use Illuminate\Database\Eloquent\JsonEncodingException;
use Illuminate\Support\Collection;
use Spatie\Activitylog\Exceptions\CouldNotLogActivity;
use Spatie\Activitylog\Facades\CauserResolver;
use Spatie\Activitylog\LogOptions;
use Spatie\Activitylog\Models\Activity;
use Spatie\Activitylog\Test\Enums\NonBackedEnum;
use Spatie\Activitylog\Test\Models\Article;
use Spatie\Activitylog\Test\Models\User;
use Spatie\Activitylog\Traits\LogsActivity;
Expand Down Expand Up @@ -431,10 +433,20 @@ public function tapActivity(Activity $activity, string $eventName)
expect($this->getLastActivity())->toBeNull();
});

// Helpers
function it_returns_an_instance_of_the_activity_after_logging()
{
$activityModel = activity()->log('test');
it('logs backed enums in properties', function () {
activity()
->withProperties(['int_backed_enum' => \Spatie\Activitylog\Test\Enums\IntBackedEnum::Draft])
->withProperty('string_backed_enum', \Spatie\Activitylog\Test\Enums\StringBackedEnum::Published)
->log($this->activityDescription);

expect($activityModel)->toBeInstanceOf(Activity::class);
}
$this->assertSame(0, $this->getLastActivity()->properties['int_backed_enum']);
$this->assertSame('published', $this->getLastActivity()->properties['string_backed_enum']);
})->skip(version_compare(PHP_VERSION, '8.1', '<'), "PHP < 8.1 doesn't support enum");

it('does not log non backed enums in properties', function () {
activity()
->withProperty('non_backed_enum', NonBackedEnum::Published)
->log($this->activityDescription);
})
->throws(JsonEncodingException::class)
->skip(version_compare(PHP_VERSION, '8.1', '<'), "PHP < 8.1 doesn't support enum");
4 changes: 4 additions & 0 deletions tests/DetectsChangesTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -1073,6 +1073,7 @@ public function getActivitylogOptions(): LogOptions
'json' => null,
'price' => null,
'interval' => null,
'status' => null,
'created_at' => '2017-01-01T12:00:00.000000Z',
'updated_at' => '2017-01-01T12:00:00.000000Z',
],
Expand Down Expand Up @@ -1117,6 +1118,7 @@ public function getActivitylogOptions(): LogOptions
'updated_at' => '2017-01-01T12:00:00.000000Z',
'user.name' => 'user name',
'interval' => null,
'status' => null,
],
];

Expand Down Expand Up @@ -1237,6 +1239,7 @@ public function getActivitylogOptions(): LogOptions
'json' => null,
'price' => null,
'interval' => null,
'status' => null,
'created_at' => '2017-01-01T12:00:00.000000Z',
],
];
Expand Down Expand Up @@ -1269,6 +1272,7 @@ public function getActivitylogOptions(): LogOptions
'user_id' => null,
'price' => null,
'interval' => null,
'status' => null,
],
];

Expand Down
10 changes: 0 additions & 10 deletions tests/Enum/NonBackedEnum.php

This file was deleted.

9 changes: 9 additions & 0 deletions tests/Enums/IntBackedEnum.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
<?php

namespace Spatie\Activitylog\Test\Enums;

enum IntBackedEnum: int
{
case Published = 1;
case Draft = 0;
}
9 changes: 9 additions & 0 deletions tests/Enums/NonBackedEnum.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
<?php

namespace Spatie\Activitylog\Test\Enums;

enum NonBackedEnum
{
case Published;
case Draft;
}
9 changes: 9 additions & 0 deletions tests/Enums/StringBackedEnum.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
<?php

namespace Spatie\Activitylog\Test\Enums;

enum StringBackedEnum: string
{
case Published = 'published';
case Draft = 'draft';
}
16 changes: 0 additions & 16 deletions tests/LogsActivitySerializationTest.php

This file was deleted.

87 changes: 87 additions & 0 deletions tests/LogsActivityTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,13 @@

use Carbon\Carbon;
use Illuminate\Database\Eloquent\SoftDeletes;
use Illuminate\Foundation\Application;
use Illuminate\Support\Collection;
use Spatie\Activitylog\LogOptions;
use Spatie\Activitylog\Models\Activity;
use Spatie\Activitylog\Test\Enums\NonBackedEnum;
use Spatie\Activitylog\Test\Models\Article;
use Spatie\Activitylog\Test\Models\ArticleWithLogDescriptionClosure;
use Spatie\Activitylog\Test\Models\Issue733;
use Spatie\Activitylog\Test\Models\User;
use Spatie\Activitylog\Traits\LogsActivity;
Expand Down Expand Up @@ -494,3 +497,87 @@ public function getActivitylogOptions(): LogOptions
$this->assertEquals('created', $this->getLastActivity()->description);
$this->assertEquals('created', $this->getLastActivity()->event);
});

it('can be serialized', function () {
$model = ArticleWithLogDescriptionClosure::create(['name' => 'foo']);

$this->assertNotNull(serialize($model));
});

it('logs non backed enum casted attribute', function () {
$articleClass = new class() extends Article {
use LogsActivity;

protected $casts = [
'status' => NonBackedEnum::class,
];

public function getActivitylogOptions(): LogOptions
{
return LogOptions::defaults()->logOnly(['status']);
}
};

$article = new $articleClass();
$article->status = NonBackedEnum::Draft;
$article->save();

$this->assertInstanceOf(get_class($articleClass), $this->getLastActivity()->subject);
$this->assertEquals($article->id, $this->getLastActivity()->subject->id);
$this->assertSame('Draft', $this->getLastActivity()->properties['attributes']['status']);
$this->assertEquals('created', $this->getLastActivity()->description);
$this->assertEquals('created', $this->getLastActivity()->event);
})->skip(
version_compare(PHP_VERSION, '8.1', '<') || version_compare(Application::VERSION, '9.0', '<'),
"PHP < 8.1 doesn't support enums && Laravel < 9.0 doesn't support non-backed-enum casting"
);

it('logs int backed enum casted attribute', function () {
$articleClass = new class() extends Article {
use LogsActivity;

protected $casts = [
'status' => \Spatie\Activitylog\Test\Enums\IntBackedEnum::class,
];

public function getActivitylogOptions(): LogOptions
{
return LogOptions::defaults()->logOnly(['status']);
}
};

$article = new $articleClass();
$article->status = \Spatie\Activitylog\Test\Enums\IntBackedEnum::Published;
$article->save();

$this->assertInstanceOf(get_class($articleClass), $this->getLastActivity()->subject);
$this->assertEquals($article->id, $this->getLastActivity()->subject->id);
$this->assertSame(1, $this->getLastActivity()->properties['attributes']['status']);
$this->assertEquals('created', $this->getLastActivity()->description);
$this->assertEquals('created', $this->getLastActivity()->event);
})->skip(version_compare(PHP_VERSION, '8.1', '<'), "PHP < 8.1 doesn't support enum");

it('logs string backed enum casted attribute', function () {
$articleClass = new class() extends Article {
use LogsActivity;

protected $casts = [
'status' => \Spatie\Activitylog\Test\Enums\StringBackedEnum::class,
];

public function getActivitylogOptions(): LogOptions
{
return LogOptions::defaults()->logOnly(['status']);
}
};

$article = new $articleClass();
$article->status = \Spatie\Activitylog\Test\Enums\StringBackedEnum::Draft;
$article->save();

$this->assertInstanceOf(get_class($articleClass), $this->getLastActivity()->subject);
$this->assertEquals($article->id, $this->getLastActivity()->subject->id);
$this->assertSame('draft', $this->getLastActivity()->properties['attributes']['status']);
$this->assertEquals('created', $this->getLastActivity()->description);
$this->assertEquals('created', $this->getLastActivity()->event);
})->skip(version_compare(PHP_VERSION, '8.1', '<'), "PHP < 8.1 doesn't support enum");
1 change: 1 addition & 0 deletions tests/TestCase.php
Original file line number Diff line number Diff line change
Expand Up @@ -80,6 +80,7 @@ protected function createTables(...$tableNames)
$table->text('json')->nullable();
$table->string('interval')->nullable();
$table->decimal('price')->nullable();
$table->string('status')->nullable();
}
});
});
Expand Down

0 comments on commit eee6143

Please sign in to comment.