Soft Deleting a Model in Laravel

Soft deleting a model in Laravel is fairly straightfoward.

You add a deleted_at column to your table using a helper method in your migration:

Schema::table('posts', function ($table) {
    ...
    $table->softDeletes();
});

Then you add the SoftDeletes trait to your model:

class Post extends Model
{
    use SoftDeletes;
...
}

That's it. When you call Post::destroy($id), it will "soft delete" your record by adding the date/time to the deleted_at column and excluding it from subsequent query results.

But what if you want to update another column too? For example, you may want to maintain a record of the logged in user in a deleted_by column. You can do this by hooking into the deleting model event (there is one caveat though).

In the example above, you would add a boot() method to the Post model and hook into the deleting model event. Here, you update the model to add the logged in user's id to the deleted_by column.

public static function boot()
{
    parent::boot();

    static::deleting(function ($post) {
        $post->deleted_by = Auth::id();
    });
}

The Caveat

When you check your table, you'll find that the deleted_by column is not populated at all! It's because the SoftDeletes trait simply updates the deleted_at column ignoring any other changes in the model itself.

In Illuminate\Database\Eloquent\SoftDeletes:

/**
 * Perform the actual delete query on this model instance.
 *
 * @return void
 */
protected function runSoftDelete()
{
    $query = $this->newQueryWithoutScopes()->where($this->getKeyName(), $this->getKey());

    $this->{$this->getDeletedAtColumn()} = $time = $this->freshTimestamp();

    $query->update([$this->getDeletedAtColumn() => $this->fromDateTime($time)]);
}

This was a bit confusing at first because when you normally hook into event like updating or creating, the changes to the model are applied when it is saved to the database.

In this case though, you have to explicitly call the save() method on the model to persist it before the soft deletion occurs. The boot() method above would look like this:

public static function boot()
{
    parent::boot();

    static::deleting(function ($post) {
        $post->deleted_by = Auth::id();
        $post->save();
    });
}

This now works. The deleted_by column is populated with the current user's id in the database and then soft deleted.

Note: This will perform an additional UPDATE query on the database. In most cases this is fine and better than overriding a framework's method.