Laravel:使用Faker

时间:2017-04-04 09:11:48

标签: php mysql laravel laravel-5.4 laravel-seeding

简介

有什么人,我对模型工厂和多个独特专栏提出了一个问题:

背景

我有一个名为Image的模型。此模型的语言支持存储在单独的模型 ImageText 中。 ImageText 有一个 image_id 列,一个语言列和一个文本列。

ImageText MySQL 中有一个约束,即 image_id 和语言必须是唯一的。

class CreateImageTextsTable extends Migration
{

    public function up()
    {
        Schema::create('image_texts', function ($table) {

            ...

            $table->unique(['image_id', 'language']);

            ...

        });
    }

    ...

现在,我希望每个图片在播种完成后都有几个 ImageText 模型。对于模型工厂和这个播种机来说这很容易:

factory(App\Models\Image::class, 100)->create()->each(function ($image) {
    $max = rand(0, 10);
    for ($i = 0; $i < $max; $i++) {
        $image->imageTexts()->save(factory(App\Models\ImageText::class)->create());
    }
});

问题

但是,当使用模型工厂和faker播种时,您经常会收到以下消息:

[PDOException]                                                                                                                 
SQLSTATE[23000]: Integrity constraint violation: 1062 Duplicate entry '76-gn' for key 'image_texts_image_id_language_unique'

这是因为在某些时候,在for循环中,faker将为图像随机选择相同的languageCode两次,从而打破了[&#39; image_id&#39;,&#39;语言&#39; ]

您可以更新 ImageTextFactory 来说明这一点:

$factory->define(App\Models\ImageText::class, function (Faker\Generator $faker) {

    return [
        'language' => $faker->unique()->languageCode,
        'title' => $faker->word,
        'text' => $faker->text,
    ];
});

但是,在您创建足够的imageTexts之后,您会遇到faker将用完languageCodes的问题。

当前解决方案

目前通过为ImageText设置两个不同的工厂来解决这个问题,其中一个工厂重置了languageCodes的唯一计数器,并且播种器调用工厂,它在进入for循环之前重置te唯一计数器以创建更多的ImageTexts。但这是代码重复,应该有更好的方法来解决这个问题。

问题

有没有办法将您保存的模型发送到工厂?如果是这样,我可以在工厂内检查当前Image是否已经附加了任何ImageTexts,如果没有,则重置languageCodes的唯一计数器。我的目标是这样的:

$factory->define(App\Models\ImageText::class, function (Faker\Generator $faker) {

    $firstImageText = empty($image->imageTexts());

    return [
        'language' => $faker->unique($firstImageText)->languageCode,
        'title' => $faker->word,
        'text' => $faker->text,
    ];
});

当然这给出了:

[ErrorException]           
Undefined variable: image

有可能以某种方式实现这一目标吗?

5 个答案:

答案 0 :(得分:3)

我解决了

我经常搜索这个问题的解决方案,并发现许多其他人也经历过这个问题。如果您只需要在关系的另一端使用一个元素it's very straight forward

添加&#34;多列独特限制&#34;是什么让这变得复杂。我发现的唯一解决方案是&#34;忘记MySQL限制,只使用针对PDO异常的try-catch围绕工厂创建&#34;。这感觉就像是一个糟糕的解决方案,因为其他PDOExceptions也会被抓住,而且它感觉不到&#34;正确&#34;。

<强>解决方案

为了完成这项工作,我将播种机分为ImageTableSeeder和ImageTextTableSeeder,它们都很直接。他们的run命令看起来像这样:

public function run()
{
    factory(App\Models\ImageText::class, 100)->create();
}

神奇发生在ImageTextFactory中:

$factory->define(App\Models\ImageText::class, function (Faker\Generator $faker) {

    // Pick an image to attach to
    $image = App\Models\Image::inRandomOrder()->first();
    $image instanceof App\Models\Image ? $imageId = $image->id : $imageId = null;

    // Generate unique imageId-languageCode combination
    $imageIdAndLanguageCode = $faker->unique()->regexify("/^$imageId-[a-z]{2}");
    $languageCode = explode('-', $imageIdAndLanguageCode)[1];

    return [
        'image_id' => $imageId,
        'language' => $languageCode,
        'title' => $faker->word,
        'text' => $faker->text,
    ];
});

就是这样:

$imageIdAndLanguageCode = $faker->unique()->regexify("/^$imageId-[a-z]{2}");

我们在regexify-expression中使用imageId,并添加我们独特组合中包含的任何内容,在这种情况下将其分隔为&#39; - &#39;字符。这将产生类似&#34; 841-en&#34;,&#34; 58-bz&#34;,&#34; 96-xx&#34;其中imageId始终是我们数据库中的真实图像,或者为null。

由于我们将唯一标记与imageId一起粘贴到语言代码中,因此我们知道 image_id和languageCode的组合将是唯一的。这正是我们所需要的!

现在我们可以通过以下方式简单地提取创建的语言代码或我们想要生成的任何其他唯一字段:

$languageCode = explode('-', $imageIdAndLanguageCode)[1];

这种方法具有以下优点:

  • 无需捕捉异常
  • 工厂和播种机可以分开以便于阅读
  • 代码紧凑

这里的缺点是你只能生成一个键组合,其中一个键可以表示为正则表达式。只要有可能,这似乎是解决这个问题的好方法。

答案 1 :(得分:2)

我根据 Rkey 的回答来满足我的需求:

问题
我有两个整数字段,它们一起应该是唯一的,它们是 product_idbranch_id

解决方案
这是我的方法:

  1. 获取产品和分支的总数。由于 id 是从 1 生成的,因此 id 的范围应从 1 到 the-total-count-of-items-in-the-table(s ).
  2. 通过创建一个由字符分隔的字符串来创建所有可能的唯一值,这些值可以从 product_idbranch_id 创建,在本例中为 -
  3. 使用 randomElements 函数从此集合中生成唯一的随机值。
  4. 将随机元素拆分回 product_idbranch_id
    $branch_count = Branch::all()->count();
    $product_count = Product::all()->count();

    $branch_products = [];
    for ($i = 1; $i <= $branch_count; $i++) {
      for ($j = 1; $j <= $product_count; $j++) {
        array_push($branch_products, $i . "-" . $j);
      }
    }

    $branch_and_product = $this->faker->unique->randomElement($branch_products);

    $branch_and_product = explode('-', $branch_and_product);
    $branch_id = $branch_and_product[0];
    $product_id = $branch_and_product[1];

    return [
      // other fields
      // ...
      "branch_id" =>  $branch_id,
      "product_id" => $product_id
    ];

答案 2 :(得分:1)

您的解决方案仅适用于可以作为组合重新使用的内容。有许多用例,多个单独的Faker生成的数字/字符串/其他对象的组合需要是唯一的,并且不能被重新编码。

对于这种情况,您可以这样做:

$factory->define(App\Models\ImageText::class, function (Faker\Generator $faker) {

    static $combos;
    $combos = $combos ?: [];

    // Pick an image to attach to
    $image = App\Models\Image::inRandomOrder()->first();
    $image instanceof App\Models\Image ? $imageId = $image->id : $imageId = null;

    // Generate unique imageId-languageCode combination
    while($languageCode = $faker->languageCode && in_array([$imageId, $languageCode], $combos) {}
    $combos[] = [$imageId, $languageCode];

    return [
        'image_id' => $imageId,
        'language' => $languageCode,
        'title' => $faker->word,
        'text' => $faker->text,
    ];
});

对于您的具体问题/用例,这里有一个解决方案:

{{1}}

答案 3 :(得分:0)

这是处理表种子类中唯一约束问题的另一种方法。

我将以一个名为 JobCategory 的模型为例。

对于JobCategory,“标题”列具有唯一的约束。

在工厂班级:

$factory->define(JobCategory::class, function (Faker $faker) {
    return [
        'title' => $faker->words(3, true),
        'description' => $faker->paragraphs(2, true),
    ];
});

然后,在播种器类中:

class JobCategoryTableSeeder extends Seeder
{
    private $failures = 0;

    /**
     * Run the database seeds.
     *
     * @return void
     */
    public function run() 
    {
        try {
            factory(JobCategory::class, 30)->create();
        } catch(Exception $e) {

            if($this->failures > 5) {
                print_r("Seeder Error. Failure count for current entity: " . $this->failures);
                return;
            }
            
            $this->failures++;
            $this->run(); // retry again until the number of failure is greater than 5
        }
    }
}

说明:

  • 这个想法是捕获可能因唯一约束失败而导致的异常,然后通过递归调用该方法直到满足退出条件来重试种子。

  • 在上面的示例中,我想创建30条记录,但是由于重试异常,我可能会获得多于或少于30条记录。

  • 我选择了5次重试,您可以使用任意数量的重试。

答案 4 :(得分:0)

我使用的是 Laravel 8.x,我不知道我使用的列函数定义是否适用于以前的版本。

我遇到了同样的问题并使用了不同的方法。

我以这种方式创建了 ImageTextFactory

<?php

namespace Database\Factories;

use App\Models\ImageText;
use Illuminate\Database\Eloquent\Factories\Factory;

class ImageTextFactory extends Factory
{
    /**
     * The name of the factory's corresponding model.
     *
     * @var string
     */
    protected $model = ImageText::class;

    /**
     * The number of models created till now.
     *
     * @var integer
     */
    protected $created = 0;

    /**
     * Define the model's default state.
     *
     * @return array
     */
    public function definition()
    {
        $this->created++;

        return [
            'language' => function (array $attributes) {
                $count = ImageText::where(
                    'image_id',
                    $attributes['image_id']
                )
                ->count();

                $reset = $this->created == 1 && $count == 0;

                return $this->faker->unique($reset)->languageCode();
            },
            'title' => $this->faker->word(),
            'text' => $this->faker->sentence(),
        ];
    }
}

然后我从播种机中调用工厂为:

Image::factory()
    ->count(10)
    ->has(
        ImageText::factory()->count(rand(0, 10))
    )->create();

使用定义中的函数,我可以检查之前是否为该 ImageText 定义了 image_id 以及生成了多少模型。当为每个 ImageTextFactory 生成一个 ImageFactory 实例时,它会自动将 $created 计数器重置为 0;并且由于 Seeder 将始终按顺序创建图像,因此它一定不会产生问题。

它有一个缺点,如果为已经存在的模型调用工厂,它将从 Faker 生成一个 OverflowException,因为没有新的 id 来重置唯一约束。它只能使用 has 方法生成。