-
Notifications
You must be signed in to change notification settings - Fork 11.3k
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
[12.x] Allowing merging model attributes before insert via Model::fillAndInsert()
#55038
Conversation
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. |
Wouldn't it be better to use: $albums = collect([
Album::make([ 'genre' => Genre::Rock ...]),
Album::make([ 'genre' => Genre::HipHop...]),
Album::make([ 'genre' => Genre::Pop...]),
])->toArray();
Album::insert($albums); |
Great call! That's another thing I have considered. I may try to add that to this PR, or maybe we'll see if it gets accepted with less moving pieces to start. 🤔 |
5beba99
to
34430da
Compare
Eloquent\Builder::insertWithCasts()
Eloquent\Builder::mergeAttributesBeforeInsert()
I do kind of wonder if some new method like that would be better than this opt-in way which feels very clunky imo. 🤔 |
I agree on the clunkiness. My original version was I'm open to any suggestions you may have @taylorotwell 🙇 |
@cosmastech hmm yeah I'm not sure I have any suggestions, but imagine a new method like |
254852e
to
06f2da3
Compare
Eloquent\Builder::mergeAttributesBeforeInsert()
Eloquent\Builder::hydrateAndInsert()
@taylorotwell after chatting with some other folks (namely Claude, Chat Gippity, and my manager), I went with
Why |
Eloquent\Builder::hydrateAndInsert()
Model::hydrateAndInsert()
* @param array<int, array<string, mixed>> $values | ||
* @return bool | ||
*/ | ||
public function hydrateAndInsert(array $values) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
thought: I believe we may want to allow creating these via collections as well. I could keep the docblock typehints and remove the parameter typehint here. Curious to know your thoughts.
@cosmastech is there an issue with just calling the method |
Hi @taylorotwell... I think that could be confusing given that the model's I don't think it would break anything to change the name here, just concerned about presenting confusion. |
@cosmastech maybe |
You're the boss, so I can make that change. However, I feel it's still perhaps confusing, since Is there a word other than Thank goodness this PR doesn't have anything to do with cache invalidation. Otherwise we'd be facing the two hardest problems in computer science simultaneously 😆 |
@cosmastech maybe |
Model::hydrateAndInsert()
Model::fillAndInsert()
@taylorotwell done. While I think a |
I submitted a PR for this about two years ago, but I wanted to take a second shot at it.
The Goal
Be able to perform a bulk insert of records without having to manually cast values to primitives, set timestamps, or set UUIDs in an array_map(). Why is this important?
Inserting models one-by-one is terribly inefficient. The
upsert()
functionality already does quite a bit of the heavy lifting (setting timestamps and unique string IDs), but doesn't play nice with casts.Current Way of Doing This
Write a custom macro. Since the last PR, I have added this to three separate projects, and will probably always add it to new projects where I need to be able to bulk insert records.
Loop through all of your records and add the casts/timestamps yourself. This works, but isn't easy to reach for.
Due to this headache, I believe it nudges users into performing inserts one-by-one, which will lead to increased DB load and performance degradations at scale. (We don't want people mistakenly thinking Laravel is slow)
How to use this functionality
This will convert the enums to their raw values, cast any objects/arrays to their DB representation, add created_at/updated_at timestamps, and set unique string IDs.