Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix model serialization bug #100

Merged
merged 1 commit into from
Dec 28, 2023

Conversation

d0m4te
Copy link
Contributor

@d0m4te d0m4te commented Dec 27, 2023

immediately apply casting to geometry objects to prevent errors while trying to json_encode binary data

fixes #55

@d0m4te
Copy link
Contributor Author

d0m4te commented Dec 27, 2023

@MatanYadaev sorry, forgot to check against phpstan.

@d0m4te d0m4te force-pushed the prevent-json_encode-error-5 branch from 7f99f56 to e3c8786 Compare December 28, 2023 00:17
immediately apply casting to geometry objects to prevent errors while trying to json_encode binary data
@d0m4te d0m4te force-pushed the prevent-json_encode-error-5 branch from e3c8786 to 8283e7d Compare December 28, 2023 00:24
Copy link
Owner

@MatanYadaev MatanYadaev left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks!

@MatanYadaev MatanYadaev merged commit c1958ae into MatanYadaev:master Dec 28, 2023
@MatanYadaev MatanYadaev changed the title prevent json_encode error 5 Fix model serialize bug Dec 28, 2023
@MatanYadaev MatanYadaev changed the title Fix model serialize bug Fix model serialization bug Dec 28, 2023
@patrickomeara
Copy link

patrickomeara commented Jan 2, 2024

This change breaks the Point casting when calling refresh() on a model.

TypeError: MatanYadaev\EloquentSpatial\Objects\Geometry::fromWkb(): Argument #1 ($wkb) must be of type string, MatanYadaev\EloquentSpatial\Objects\Point given, called in /Users/pat/Code/web/vendor/matanyadaev/laravel-eloquent-spatial/src/GeometryCast.php on line 47
/Users/pat/Code/web/vendor/matanyadaev/laravel-eloquent-spatial/src/Objects/Geometry.php:72
/Users/pat/Code/web/vendor/matanyadaev/laravel-eloquent-spatial/src/GeometryCast.php:47
/Users/pat/Code/web/vendor/laravel/framework/src/Illuminate/Database/Eloquent/Concerns/HasAttributes.php:816
/Users/pat/Code/web/vendor/laravel/framework/src/Illuminate/Database/Eloquent/Concerns/HasAttributes.php:792
/Users/pat/Code/web/vendor/laravel/framework/src/Illuminate/Database/Eloquent/Concerns/HasAttributes.php:2129
/Users/pat/Code/web/vendor/laravel/framework/src/Illuminate/Database/Eloquent/Concerns/HasAttributes.php:489
/Users/pat/Code/web/vendor/laravel/framework/src/Illuminate/Database/Eloquent/Concerns/HasAttributes.php:443
/Users/pat/Code/web/vendor/matanyadaev/laravel-eloquent-spatial/src/Traits/HasSpatial.php:27
/Users/pat/Code/web/vendor/laravel/framework/src/Illuminate/Database/Eloquent/Model.php:1697

It looks like the casting is happening twice and the second time via getAttribute() breaks the type.

@MatanYadaev
Copy link
Owner

Hi @patrickomeara , can you please submit a PR that fixes it?

@jbajou
Copy link

jbajou commented Jan 2, 2024

we're impacted by this, had to downgrade

@MatanYadaev
Copy link
Owner

Please provide examples on how to reproduce this.

@patrickomeara
Copy link

@jbajou Are you able to send me a code snippet that fails for you? I will add some failing tests and find a solution.

@jbajou
Copy link

jbajou commented Jan 3, 2024

Well, to be honest it's quite simple. We have a Location model

class Location extends Model
{
    use hasSpatial;
    
    protected $casts = [
        'point' => Point::class
    ];
}

Then, in a command (run on AWS SQS, with vapor) we do a updateOrCreate:

Location::query()
...
->updateOrCreate([
    ...
], [
    'point'           => $centralLocation->point,
]);

Here is the stack trace (partial):

TypeError MatanYadaev\EloquentSpatial\Objects\Geometry::fromWkb(): Argument #1 ($wkb) must be of type string, MatanYadaev\EloquentSpatial\Objects\Point given, called in /var/task/vendor/matanyadaev/laravel-eloquent-spatial/src/GeometryCast.php on line 47 
    vendor/matanyadaev/laravel-eloquent-spatial/src/Objects/Geometry.php:72 MatanYadaev\EloquentSpatial\Objects\Geometry::fromWkb
    vendor/matanyadaev/laravel-eloquent-spatial/src/GeometryCast.php:47 MatanYadaev\EloquentSpatial\GeometryCast::get
    vendor/laravel/framework/src/Illuminate/Database/Eloquent/Concerns/HasAttributes.php:816 Illuminate\Database\Eloquent\Model::getClassCastableAttributeValue
    vendor/laravel/framework/src/Illuminate/Database/Eloquent/Concerns/HasAttributes.php:792 Illuminate\Database\Eloquent\Model::castAttribute
    vendor/laravel/framework/src/Illuminate/Database/Eloquent/Concerns/HasAttributes.php:2129 Illuminate\Database\Eloquent\Model::transformModelValue
    vendor/laravel/framework/src/Illuminate/Database/Eloquent/Concerns/HasAttributes.php:489 Illuminate\Database\Eloquent\Model::getAttributeValue
    vendor/laravel/framework/src/Illuminate/Database/Eloquent/Concerns/HasAttributes.php:443 Illuminate\Database\Eloquent\Model::getAttribute
    vendor/matanyadaev/laravel-eloquent-spatial/src/Traits/HasSpatial.php:27 App\Models\Location::setRawAttributes

@shengslogar
Copy link

Same stack trace over here.

Culprit line:

3.2.0...3.2.1#diff-8fa4ecc9982e0985829510274a951b022588e8be8c90e6fca5cbe23eb7edab8eR27

I don't fully understand the underlying implementation of or rationale behind this change, but obvious fix is updating GeometryCast to pass-through already-casted values.

Guessing the reason this was never encountered before is because Laravel caches casted values by default. Overriding setRawAttributes feels dangerous.

@MatanYadaev
Copy link
Owner

Hi @jbajou, your example doesn't appear to reproduce the bug.

I'm going to revert the latest changes to setRawAttributes, I know it means the serialization issue will return, but as other people said here, overriding setRawAttributes is not a good practice. I'll be happy to accept PRs and ideas on how to overcome this issue with a more elegant solution.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Unable to JSON encode payload. Error code: 5
5 participants