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

[11.x] Introduce freezing Model #53472

Draft
wants to merge 7 commits into
base: 11.x
Choose a base branch
from

Conversation

cosmastech
Copy link
Contributor

@cosmastech cosmastech commented Nov 11, 2024

In this PR, we introduce the concept of a frozen model.

A frozen model is disallowed to do the following:

  • Fill attributes
  • Set attributes
  • Load new relations
  • Update
  • Touch
  • Save, SaveQuietly
  • Delete
  • Increment, incrementQuietly
  • decrement, decrementQuietly
  • pre-emptively freeze loaded relations
  • disallow loading on Eloquent\Collection
  • disallow "fresh" on Eloquent\Collection
  • other Eloquent\Collection concerns?

Basically, the model becomes an inert object which disallows fetching relations and modifying the underlying table.

Why?

The goal here is to allow passing models as "dumb objects." One of the biggest pains I have experienced is models get passed to resources and or actions/services and end up performing side effects. These side effects can cause things like N+1 problems, plus hard to follow/maintain logic. The lazy-loading violation kind of helps us here, but it has the weird gotcha that it can still load relations if it was loaded singly.

A frozen model communicates intent: you can look, but you don't touch.

Alternatives approaches as of today

Map the model to a data object and pass that around instead. I think that's honestly probably the best, since they are inert. But mapping from a model to a DTO can bring its own headaches.

Alternate approaches to this PR

Create a FrozenModel class This would quite possibly be better, as it could leverage strong type-hints, but would only be able to indicate the model is frozen via a template. Something like:
/**
 * @template TModel of Model
 */
class FrozenModel
{
    /** @var TModel $model */
    protected Model $model;

    public function __construct(Model $model)
    {
        $this->model = $model;
    }

    public function __call($method, $parameters)
    {
        if (in_array($method, ['save', 'fill', 'update', 'touch', 'loadMissing', /* ... */])) {
            throw new FrozenModelException("Cannot call [$method] on frozen model[$model->getKey()]");
        }
        return $this->model->__call($method, $parameters);
    }
}

and give the model class a simple method.

class Model // ...
{
    /** all the use stuff */

   /**
    * @return FrozenModel<$this>
    */
   public function freeze(): FrozenModel
   {
       return new FrozenModel($this);
   }
}

As I'm writing the description, I think I kind of prefer this approach, but figured I'd float out an idea and get maintainer/community feedback.

I do realize while this is a draft, there won't be a formal review.

Copy link

Thanks for submitting a PR!

Note that draft PR's are not reviewed. If you would like a review, please mark your pull request as ready for review in the GitHub user interface.

Pull requests that are abandoned in draft may be closed due to inactivity.

@josephkerkhof
Copy link

I like this idea. What if the Immutable Object Pattern is used to achieve this effect? It should be familiar to Laravel developers. DateTime and Carbon objects follow this pattern.

You could take an instance of a model and do a ->toImmutable() on it.

@cosmastech cosmastech force-pushed the model-freeze branch 2 times, most recently from e7012d4 to c697b39 Compare November 12, 2024 02:47
@calebdw
Copy link
Contributor

calebdw commented Nov 14, 2024

The frozen state needs to be specifiable and enforceable by the type system / static analysis and not a runtime property (think a separate class like Carbon/CarbonImmutable, or a intersection type like Model&Immutable) ---otherwise every model mutation would need to be guarded by if (! $model->isFrozen()). That's not to say that you can't throw runtime exceptions for invalid mutations, but the type system needs to know if models are writable.

Also note that the CarbonImmutable instance allows changes, it just returns a new static instance instead of mutating the original---this is different from throwing exceptions when attempting to change the instance.

If a model becomes frozen, does this also recursively freeze all loaded relations? I would expect so as I would not want to pass a frozen model to a consumer and then have them mutate a related model.

Some other methods to potentially look at restricting are: __set/offsetSet, __call, all of the load* methods, save, {re}fresh, all other set* methods, etc.

I like the idea because I don't necessarily care to create DTOs that basically contain the same properties, just some food for thought.

Best,

@cosmastech
Copy link
Contributor Author

Thanks for the reply @calebdw!

The frozen state needs to be specifiable and enforceable by the type system / static analysis and not a runtime property (think a separate class like Carbon/CarbonImmutable, or a intersection type like Model&Immutable) ---otherwise every model mutation would need to be guarded by if (! $model->isFrozen()). That's not to say that you can't throw runtime exceptions for invalid mutations, but the type system needs to know if models are writable.

I've been thinking about this a lot. I hadn't considered the Immutable marker interface as an option, so thank you for that.

The problem I see with most of the non-runtime solutions is that I don't see a way to avoid some kind of code generation.

Immutable marker interface

Take for instance some code like this:

public function loginAs(User&Immutable $user): void
{
  // ...
}

loginAs(User::first()->toFrozen()); // ❌ User does not implement `Immutable`

There would be two options:

  • Every Model implements Immutable, but then the type-safety benefit is lost. (ie: any time I pass User I'm passing User&Immutable)
  • Return an anonymous class:
class User extends Model
{
    public function toFrozen(): User&Immutable
    {
        return new class extends User implements Immutable {
            // some logic here for constructing the object
        };
    }
}

Problem is that then Models cannot be marked as final. Maybe this isn't a big deal. Also, anonymous classes are kind of ugly to debug when things break.

I don't see a way to get a leverage the PHP type system without creating a separate class (probably employing code generation to do so). At that point, it seems like this would fall better in line with something like spatie's laravel-data than it would within the framework itself.

Create a FrozenModel class

/**
 * @param FrozenModel<User> $user
 */
public function loginAs(FrozenModel $user) { /* ... */ }
loginAs(User::first()->toFrozen()); // ✅ by the type-system && static analysis
loginAs(Team::first()->toFrozen()); // ✅ by the type-system, but ❌  fails by static analysis.

The benefit is only here if we enforce static analysis.

A real-time Immutable object

The other option is to do something like real-time facades, but I found the support for that was wanting in PHPStorm & Laravel IDEA (though I think that may have changed recently). It also looks like larastan doesn't support them. This may have the same problems with marking a class as final, haven't thought about it fully.


If a model becomes frozen, does this also recursively freeze all loaded relations? I would expect so as I would not want to pass a frozen model to a consumer and then have them mutate a related model.

I forgot about that. Good call. 👍 I have added it to the checklist.

I like the idea because I don't necessarily care to create DTOs that basically contain the same properties, just some food for thought.

I'm glad you think it could be valuable as well. Part of me just doesn't want to spend the time writing this without some guidance on which direction the maintainers/community would prefer, or if the idea is totally unacceptable. It's probably another 4+ hours of dev with any of these approaches.

@juampi92
Copy link
Contributor

Hey,

My two cents on this PR: while the idea is promising, more evidence is needed for the framework to maintain this as a core feature. The way to prove this need is by creating a package. Packages are meant to be a choose-your-own-adventure where you tweak the framework until it does exactly what you want. If the community appreciates the feature and understands its purpose, it could eventually become a first-party feature.

There are more limitations when you can't edit the framework's source code, so you will need to find other strategies to achieve the same thing, which can be more challenging. If it eventually becomes incorporated by the framework as a first-party feature, it will be integrated with the core features, which overall increase the complexity of maintaining the framework.

I suggest starting with a static analysis approach (like a frozen model with generics) because it’s the least intrusive, and this feature can't be a runtime check. If you want to sell this idea further, PHPStan/Pest architectural rules to make sure JSON Resources are not using mutable models would make this easy to adopt and benefit immediately.

The point I'm trying to make is that to integrate something into the framework, you need to build a strong case for why it’s needed and why many people will benefit from it. Making a PR on the framework will already grant you a lot of exposition to the community, so I understand the benefits of making this PR, but it's not the way these ideas are usually tested.

It's good to get this idea started, I wish we had a place in our Laravel community to do RFCs for packages :/

@cosmastech
Copy link
Contributor Author

It's good to get this idea started, I wish we had a place in our Laravel community to do RFCs for packages :/
I wish there was a sort of RFC process for Laravel proposals. I know it's a bummer to spend many hours writing something and having it closed.

I'm leaning towards closing this PR, as I think the likelihood of this getting accepted with the massive number of changes is probably close to 0. I may pivot and try just taking the static analysis approach, because like you said, it's less invasive.

@calebdw
Copy link
Contributor

calebdw commented Nov 18, 2024

This Laravel news article looks interesting / related:

https://laravel-news.com/zero-to-prod-data-model-php-package

@bulletproof-coding
Copy link

bulletproof-coding commented Dec 10, 2024

@cosmastech I like the idea. To me the ->toArray() will do this job just fine but in order to have some kind of DTO, maybe a child model could be used that will overwrite the parent "set" functions and which should be hydrated on construct only. The child could even be an anonymous class that extends the model's class (because it could already be a child of Model).

Inspired by this

https://github.com/macropay-solutions/laravel-crud-wizard-free/blob/production/src/Models/Attributes/BaseModelAttributes.php

Another class that implements only the __get() function could also be used.

@bulletproof-coding
Copy link

bulletproof-coding commented Dec 10, 2024

@cosmastech
I managed to make a demo coupled with the lib from above that I use that has the advantage of autocomplete for the attributes without creating an actual DTO.

In my project I have the above Attribute class filled like in this demo https://github.com/macropay-solutions/laravel-crud-wizard-demo/blob/production/app/Models/Attributes/OperationAttributes.php

In the Operation model I added:

    public function getFrozen(): OperationAttributes
    {
        return new class ($this) extends OperationAttributes {
            public function __set(string $key, mixed $value): void
            {
            }

            public function __unset(string $key): void
            {
            }

            public function __call(string $method, array $parameters): mixed
            {
                return null;
            }
        };
    }

Of course the anonymous class can be removed by following this logic https://github.com/macropay-solutions/laravel-crud-wizard-free/blob/670949aeb85f4067825838e86569bd649ed2c708/src/Models/BaseModel.php#L51

BaseModel

    public function getFrozen(): ?BaseModelFrozenAttributes
    {
        $frozenAttributes = \substr($class = static::class, 0, $l = (-1 * (\strlen($class) - \strrpos($class, '\\') - 1))) .
            'Attributes\\' . \substr($class, $l) . 'FrozenAttributes';

        return \class_exists($frozenAttributes) ? new $frozenAttributes((clone $this)->forceFill($this->toArray())) : null;
    }

Operations model

    public function getFrozen(): ?OperationFrozenAttributes
    {
        return parent::getFrozen();
    }

Of course the above 2 can be merged in only one in the Operation model.

    public function getFrozen(): OperationFrozenAttributes
    {
        return new OperationFrozenAttributes((clone $this)->forceFill($this->toArray())); 
        // this will include also the loaded relations
    }
<?php


/**
 * For properties autocompletion declare in the children classes (with @ property) all the model's parameters (columns)
 *  or @mixin BaseModelAttributesChild
 */
class BaseModelFrozenAttributes
{
    public function __construct(
        private BaseModel $ownerBaseModel
    ) {
    }

    public function __get(string $key): mixed
    {
        return $this->ownerBaseModel->getAttribute($key);
    }

    public function __isset(string $key): bool
    {
        return $this->ownerBaseModel->__isset($key);
    }

    public function toArray(): array
    {
        return $this->ownerBaseModel->toArray();
    }
}


/**
 * @property int $id
 * @property ?int $parent_id
 * @property int client_id
 * @property string currency
 * @property string value
 * @property ?string created_at
 * @property ?string updated_at
 * or
 * @mixin OperationAttributes
 */
class OperationFrozenAttributes extends BaseModelFrozenAttributes
{
}

In both cases the autocomplete works on the frozen class while reading from the actual model.

    /** @var Operation $operation */
    $operation = Operation::query()->firstOrFail();
    $frozenModel = $operation->getFrozen();
    // $frozenModel->{autocomplete here for attributes declared with @property};
    // $frosenModel->relationName will be array in this case or null and will be autocompleted only if declated

@macropay-solutions see this.

@macropay-solutions
Copy link

macropay-solutions commented Dec 11, 2024

@marius-mcp
Thank you.
We created a draft here macropay-solutions/laravel-crud-wizard-free#8 and demo macropay-solutions/laravel-crud-wizard-demo#2
See the difference in toArray function. Attributes contain already the relations so, the toArray makes sense to retrieve only the attributes.

@cosmastech Nice idea.

Update
we transformed toArray funciton into __toString that returns json to ease the @mixin usage for avoiding de duplicate declarations of properties.

Update
We realized that Reflection or closure bindings could give access tot he private model from the Frozen Attributes class so we changed it to \stdClass in here.
This will also allow for fluent access like

     $frozen->oneToOneRelationName->relationColumn;

@shaedrich
Copy link
Contributor

@juampi92 Taylor shout use this as a closing reason reply template 👍

@macropay-solutions
Copy link

macropay-solutions commented Dec 17, 2024

Updated the PR to forbid reflection or closure bindings to access the model.

Updated again: Make the stdClass read only also for its content and update README.md
Will continue this conversation on this dicussion.

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.

7 participants