Heim > Fragen und Antworten > Hauptteil
Also führe ich einige Unit-Tests durch, bevor ich neue Funktionen implementieren. Ich führe den Test aus und er schlägt fehl mit OverflowException :最大重试次数达到 10000 次,但未找到唯一值
Dies ist der Test, den ich durchführe.
public function test_job_factory() { $client = Client::factory()->create(); $accountHandler = AccountHandler::factory()->create(); $user = User::factory()->create(); $this->post('/login', [ 'email' => $user->email, 'password' => 'password', ]); $user->givePermissionTo( 'manage jobs' ); $clientContact = ClientContact::factory()->create(); $job = Job::factory()->create(); $this->assertTrue($job->id > 0); }
Der Fehler scheint beim Erstellen des Jobs selbst aufzutreten. Der obige Test hat andere Pflanzen getestet und funktioniert.
Dies ist die JobFactory.php-Datei:
<?php namespace Database\Factories; use App\Models\AccountHandler; use App\Models\Client; use App\Models\ClientContact; use App\Models\Job; use App\Models\User; use Illuminate\Database\Eloquent\Factories\Factory; /** * @extends \Illuminate\Database\Eloquent\Factories\Factory<\App\Models\Job> */ class JobFactory extends Factory { protected $model = Job::class; /** * Define the model's default state. * * @return array<string, mixed> */ public function definition() { return [ 'title' => $this->faker->company(), 'is_active' => 1, 'is_draft' => 0, 'client_id' => $this->faker->unique()->numberBetween(1, Client::count()), 'account_handler_id' => $this->faker->unique()->numberBetween(1, AccountHandler::count()), 'eclipse_contact_id' => $this->faker->unique()->numberBetween(1, User::count()), 'client_contact_id' => $this->faker->unique()->numberBetween(1, ClientContact::count()), 'description' => $this->faker->paragraphs(1), ]; } }
Und die Migration dafür (create_jobs_table.php):
<?php use Illuminate\Database\Migrations\Migration; use Illuminate\Database\Schema\Blueprint; use Illuminate\Support\Facades\Schema; return new class extends Migration { /** * Run the migrations. * * @return void */ public function up() { Schema::create('jobs', function (Blueprint $table) { $table->id('number'); $table->boolean( 'is_active' )->default(true); $table->boolean( 'is_complete' )->default(true); $table->string('title', 64)->nullable(); $table->timestamps(); $table->foreignId('author')->nullable()->constrained()->references('id')->on('users'); $table->text('description')->nullable(); $table->foreignId('client_id')->nullable()->constrained()->references('id')->on('clients'); $table->foreignId('client_contact_id')->nullable()->constrained()->references('id')->on('client_contacts'); $table->foreignId('account_handler_id')->nullable()->constrained()->references('id')->on('account_handlers'); $table->date('expiry_date')->nullable(); $table->date('artwork_deadline')->nullable(); $table->date('proof_deadline')->nullable(); $table->integer('redirect')->nullable(); $table->boolean( 'is_draft' )->default(true); $table->foreignId('eclipse_contact_id')->nullable()->constrained()->references('id')->on('users'); $table->foreignId('subscription_id')->nullable()->constrained()->references('id')->on('subscriptions'); }); } /** * Reverse the migrations. * * @return void */ public function down() { Schema::dropIfExists('jobs'); } };
Was ist also schief gelaufen und warum kommt es zu diesem Zyklus? Ich habe die zum Erstellen des Jobs erforderlichen Mindestdaten hinzugefügt und bin mir daher nicht sicher, was mir fehlt.
Danke
*** Bearbeiten *** Ich wurde gebeten, einen Link bereitzustellen, wo ich die schlechte Praxis gefunden habe, Unique und NumberBetween zu verwenden. Hier ist ein Beispiel. Das wird nicht so funktionieren, wie Sie es suchen!
P粉4425761652024-04-07 10:22:31
问题是当你尝试获取 1 到 5 之间的 5 个值时
(包括),你只能取4个。第5个数字永远不会超过
unique()
验证。
这是因为,例如,当您运行 unique 的第一个实例时
$faker->unique()->numberBetween(1, 20)
然后在另一个工厂中运行它,laravel 通常会扩展前一个实例(我猜),如果这有意义的话。 但是当你像下面这样传递true时
$faker->unique(true)->numberBetween(1, 20)
它再次开始从 1 到 20 搜索
这解决了我的问题。
因此始终在 unique()
实例中传递 true
P粉1649427912024-04-07 00:42:24
我建议遵循文档方法来解决此问题。 https://laravel.com/docs/9.x/database-测试#工厂关系
看看你的迁移表,你的工厂很有可能无法工作,因为你在SQL级别上指示了关系,这基本上会迫使你在相关表中拥有记录,否则SQL会抛出错误..