Laravel:有效地检索多态属性

时间:2015-06-17 08:09:43

标签: rest laravel polymorphism eloquent

我的实际问题是: 如何将属性返回给我实际需要的客户端,将其余部分抛弃并尽可能高效?

所以,我有一个构造,我基本上有媒体对象,可以是不同的类型,如视频,图片,链接等。这是以多态方式构造的。在这个例子中,用户可以拥有一个虚拟形象,实际上是一个媒体对象,但我唯一对客户端感兴趣的是图像网址。

我们的应用程序是RESTful,如果我想列出所有用户,我希望用户JSON基本上返回一个属性 only avatar的url而不是完整的多态模型,然后我必须在客户端遍历,更不用说从服务器发送到客户端的巨大开销。

来自用户模型:

/**
 * A user can have an avatar
 *
 * @return \Illuminate\Database\Eloquent\Relations\MorphOne
 */
public function avatar() {
    return $this->morphOne('Fanlete\Media', 'appendable');
}

来自媒体模型:

/**
 * A media element can be extended in different ways (kind of inheritance), this function morphs to the specific implementation
 *
 * @return \Illuminate\Database\Eloquent\Relations\MorphTo
 */
public function mediable()
{
    return $this->morphTo();
}

/**
 * A media element can be appended by various models, i.e. a user to save its avatar
 *
 * @return \Illuminate\Database\Eloquent\Relations\MorphTo
 */
public function appendable() {
    return $this->morphTo();
}

从图片模型,具体的媒体实施:

public function media()
{
    return $this->morphMany('Fanlete\Media', 'mediable');
}

所以,让我们说我有一个功能来列出我控制器中的所有用户:

public function index()
{
    return User::all();

}

我想要这样的输出:

{
    "id": 1,
    "email": "user@example.org",
    "name": "Example",
    "surname": "User",
    "created_at": "2015-06-17 07:05:26",
    "updated_at": "2015-06-17 07:05:26",
    "avatar": "http://upload.wikimedia.org/wikipedia/en/3/38/Avatarjakeneytiri.jpg"
}

目前我这样解决了这个问题,感觉实际上并不那么优雅,而且非常错误和瓶颈:

再次

用户模型:

protected $appends = ['avatar'];

public function getAvatarAttribute()
{
    return $this->avatar()->get()->first()->mediable()->get()->first()->url;
}

当然我也可以在我的控制器中做这样的事情:

$users = Users::with("avatar.mediable")->get()->all();
// Now sort out attributes I don't need and return array

并且没有sortin输出看起来像这样(非常混乱和巨大的开销):

{

    "id": 1,
    "email": "user@example.org",
    "name": "Example",
    "surname": "User",
    "created_at": "2015-06-17 07:05:26",
    "updated_at": "2015-06-17 07:05:26",
    "avatar": 

    {
        "id": 6,
        "appendable_id": 1,
        "appendable_type": "Fanlete\\User",
        "mediable_id": 6,
        "mediable_type": "Fanlete\\Picture",
        "created_at": "2015-06-17 07:05:26",
        "updated_at": "2015-06-17 07:05:26",
        "mediable": 

                {
                    "id": 6,
                    "url": "http://upload.wikimedia.org/wikipedia/en/3/38/Avatarjakeneytiri.jpg",
                    "title": null,
                    "created_at": "2015-06-17 07:05:26",
                    "updated_at": "2015-06-17 07:05:26"
                }
            }
}

但这也可能是一种解决方法,而且正是我不需要和想要的开销。

THX

1 个答案:

答案 0 :(得分:0)

你可能就在那里。

你的getAvatarAttribute()确实太复杂了。您可以尝试执行以下操作:

/** we hide the relation from the json */
protected $hidden = ['avatar'];

/** we append the url of the avatar to the json */
protected $appends = ['avatar_url'];

/**
 * To not be confused with the avatar() method, I used avatar_url.
 */
public function getAvatarUrlAttribute()
{
    return $this->avatar->mediable->url;
}

然后,您应该能够在没有瓶颈的情况下执行$users = User::with('avatar.mediable')->get();,并获得良好的JSON响应。