如何在雄辩模型中创建复杂虚拟列?

时间:2019-02-06 14:36:55

标签: php mysql laravel eloquent

这个问题与this one有关,但是我简化了我的实际问题。因此,我创建了一个单独的问题:

我有两个桌子。一个保持间隔(以天为单位)与bimonthly = 15之类的名称相关联,另一个保持间隔列出周期性动作并与间隔相关联。

我想列出在纯SQL中容易实现的下一个周期性操作:

SELECT *, DATE_ADD(updated_at, INTERVAL days DAY) AS next_action 
FROM periodic_actions 
INNER JOIN intervals ON interval_id = intervals.id
HAVING next_action > NOW()

这是我的桌子:

CREATE TABLE `intervals` (
  `id` INT NOT NULL AUTO_INCREMENT,
  `name` VARCHAR(45) NULL,
  `days` INT NULL,
  PRIMARY KEY (`id`));

CREATE TABLE `periodic_actions` (
  `id` INT NOT NULL AUTO_INCREMENT,
  `interval_id` INT NOT NULL,
  `name` VARCHAR(45) NULL,
  `updated_at` TIMESTAMP NOT NULL,
  PRIMARY KEY (`id`));

间隔的内容:

+----+----------+------+
| id | name     | days |
+----+----------+------+
|  1 | daily    |    1 |
|  2 | monthly  |   30 |
|  3 | annually |  365 |
+----+----------+------+

我想用Eloquent构建相同的查询,在其中我可以在虚拟列where上使用next_action

Action::orderBy('next_action', 'asc')
    ->where('next_action', '>', Carbon::now())
    ->get()

我无法隐藏到模型中的最接近的是:

Action::join('intervals', 'interval_id', '=', 'intervals.id')
    ->select(['*', DB::RAW('DATE_ADD(updated_at, INTERVAL days DAY) AS next_action')])
    ->havingRaw(DB::RAW('next_action > NOW()'))->get();

我当然有这个基本模型:

class Action extends Model
{
    protected $table = 'periodic_actions';

    public function interval() 
    {
        return $this->belongsTo(Interval::class);
    }
}

1 个答案:

答案 0 :(得分:2)

解决方案1:使用全局范围将该字段添加到您的选择语句中

关于虚拟列的一个问题是Eloquent,如果您不阻止,则会尝试更新它们。您可以将其从$ fillable数组中排除,并在模型中使用global scope添加虚拟列,例如:

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

    static::addGlobalScope('next_action', function (Builder $builder) {
        $builder->addSelect(DB::raw('DATE_ADD(updated_at, INTERVAL days DAY) AS next_action'));
    });
}

您必须记住,这会将您的Model耦合到MySQL,因为您使用的日期函数在sqlite之类的数据库中不起作用。

解决方案2:使用MySQL视图

通常,当我有多个计算字段时,我要做的是创建一个MySQL视图。使用所需的所有计算字段创建视图,然后可以为该视图创建新的Eloquent模型,而不必担心查询范围。

一个警告是,您当然不能在此模型上使用create / update / delete,但是如果您仅将模型用于显示目的,那应该没问题。