I am trying to implement the Factory Design pattern with a Laravel Model. I have a Reward Model wich can have (for now) two types, the Points reward or the cashback reward.
I have the class Reward with everything
class Reward extends Model
{
// class Reward stuff
}
And I also made two classes extending Reward (bboth of them with a globalScope refering to its type), PointsReward:
class PointsReward extends Reward
{
protected $table = "rewards";
protected static function boot()
{
parent::boot();
static::addGlobalScope(
'points',
function (Builder $builder) {
$builder->whereHas(
'rewardType',
function ($subQuery) {
$subQuery->where('type', 'Points');
}
);
}
);
}
}
and CashbackReward:
class CashbackReward extends Reward
{
protected $table = "rewards";
protected static function boot()
{
parent::boot();
static::addGlobalScope(
'cashback',
function (Builder $builder) {
$builder->whereHas(
'rewardType',
function ($subQuery) {
$subQuery->where('type', 'Points');
}
);
}
);
}
}
Now I was trying to made a Factory class:
class RewardFactory
{
const CASHBACK_REWARD = 0;
const POINTS_REWARD = 1;
static function create(int $rewardType, array $data)
{
switch($rewardType){
case self::CASHBACK_REWARD:
return CashbackReward::create($data);
break;
case self::POINTS_REWARD:
return PointsReward::create($data);
break;
default:
throw new Exception("Not supported");
}
}
static function find(int $id) : Reward
{
$main = Reward::find($id);
switch($main->rewardType->type){
case 'Points':
return PointsReward::find($id);
break;
case 'Cashback':
return CashbackReward::find($id);
break;
default:
throw new Exception("Not supported");
return null;
}
}
}
My problem is on the find method. Its purpose is to find a Reward by id (like Laravel method) and return an object depending on its rewardType->type
property.
For sure this works, but I have to do 2 db queries to instantiate the object. Is there a better approach for this? If no, Is there a way to do this with only one query?
via Chebli Mohamed
Aucun commentaire:
Enregistrer un commentaire