【问题标题】:How with laravel factory add data to related table?laravel 工厂如何将数据添加到相关表中?
【发布时间】:2021-08-15 07:55:51
【问题描述】:

在 laravel 8 应用程序中,我为带有工厂的 ads 表添加了虚拟数据:

$ads = Ad::factory()->count(10)->expired($year, $month)->create([
]);

还有数据库/工厂/AdFactory.php:

<?php

namespace Database\Factories;

use Config;
use Carbon\Carbon;
use App\Models\Ad;
use Illuminate\Database\Eloquent\Factories\Factory;
use Illuminate\Support\Str;
use \Cviebrock\EloquentSluggable\Services\SlugService;

class AdFactory extends Factory
{
    protected $model = Ad::class;

    public function definition()
    {
        $text= $this->faker->text;
        $slugValue = SlugService::createSlug(Ad::class, 'slug', $text);

        return [
            'title' => $text,
            'ad_token' => $text,
            'slug' => $slugValue,
            'phone_display' => (rand(1, 3) == 1),
            'has_locations' => (rand(1, 4) == 1),
            'status' => 'A',
            'price' => mt_rand(10, 500),
            'ad_type' => (rand(1, 2) == 1 ? 'B' : 'S'),
            'description' => $this->faker->paragraphs(rand(1, 4), true),
            'creator_id' => rand(1, 5),
        ];
    }

    public function expired($year, $month)
    {
        return $this->state(function (array $attributes) use($year, $month) {
            $dateStr= $year.'-'.str_pad($month, 2, "0", STR_PAD_LEFT).'-01';
            $startDate= Carbon::createFromFormat('Y-m-d', $dateStr);
            return [
                'expire_date' => $this->faker->dateTimeInInterval($startDate, '1 month', Config::get('app.timezone'))->format('Y-m-d H:i:s')
            ];
        });
    }
}

它工作正常,但我想知道如何将虚拟数据添加到相关的 ad_categories(有 ad_id 字段)中? 定义方法返回广告对象(带有新 id),我在哪里可以访问它?

谢谢!

【问题讨论】:

    标签: laravel laravel-factory


    【解决方案1】:

    您可以创建一个 CategoryFactory 类并创建 Dummy 类别,然后选择一个随机类别作为 category_id

    对于一对多关系

        //...
        public function definition()
        {
            return [
                //your code
                'creatory_id' => $this->faker->randomElement(Category::all())['id'],
            ];
        }
        //...
    

        public function definition()
        {
            return [
                //your code
                'creatory_id' => factory(Category::class)->create()->id,
            ];
        }
    

    第二种解决方案将为每个广告创建一个新类别

    对于多对多关系

    namespace Database\Factories;
    
    use App\Models\AdFactory;
    use App\Models\Category;
    use Illuminate\Database\Eloquent\Factories\Factory;
    use Illuminate\Support\Str;
    
    class AdFactory extends Factory
    {
        /**
         * The name of the factory's corresponding model.
         *
         * @var string
         */
        protected $model = Ad::class;
    
        /**
         * Configure the model factory.
         *
         * @return $this
         */
        public function configure()
        {
            return $this->afterCreating(function (Ad $newAd) { 
                $adCategory = AdCategory::factory()->count(1)->create([ 'ad_id' => $newAd->id, 'category_id' => $this->faker->randomElement(Category::all())['id']); 
            }); 
        }
    
        // ...
    }
    

    欲了解更多信息,请访问https://laravel.com/docs/8.x/database-testing#factory-callbacks

    https://laravel.com/docs/8.x/database-testing#polymorphic-many-to-many-relationships

    【讨论】:

    • 你的例子不清楚。什么是“creatory_id”?我将详细说明:对于 ads 表,我还有带有 id、name 字段的类别表。我需要将虚拟数据添加到 ad_categories(字段 ad_id、category_id)中,其中 ad_id - 参考新创建的广告,category_id - 参考现有类别之一。谢谢!
    • 哦,我以为是一对多的关系。但您可以将广告与随机虚拟类别相关联
    • 抱歉,不清楚。酷你请举个例子?而且我无法理解如何获取和使用 NewlyAd.id ?
    • 谢谢!我修复了你的代码: return $this->afterCreating(function (Ad $newAd) { // 此处返回广告模型 $adCategory = AdCategory::factory()->count(1)->create([ 'ad_id' => $newAd->id, 'category_id' => $this->faker->randomElement(Category::all())['id'], ]); });请编辑它,我会接受你的回答!
    • @mstdmstd 完成!
    猜你喜欢
    • 2018-05-01
    • 1970-01-01
    • 2020-03-22
    • 1970-01-01
    • 2015-12-09
    • 1970-01-01
    • 2019-11-11
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多