id(); $table->uuid('uuid')->unique(); $table->foreignId('program_id')->constrained('programs')->cascadeOnDelete(); $table->foreignId('participant_id')->constrained('participants')->cascadeOnDelete(); $table->foreignId('certificate_template_id')->nullable()->constrained('certificate_templates')->nullOnDelete(); $table->string('certificate_no', 100)->unique()->nullable(); $table->string('file_path', 500)->nullable(); $table->string('token', 64)->unique(); $table->enum('status', ['pending', 'generating', 'generated', 'emailed', 'failed'])->default('pending'); $table->text('error_message')->nullable(); $table->timestamp('generated_at')->nullable(); $table->timestamp('emailed_at')->nullable(); $table->timestamp('downloaded_at')->nullable(); $table->unsignedInteger('download_count')->default(0); $table->timestamps(); $table->unique(['program_id', 'participant_id']); $table->index('token'); $table->index('status'); $table->index('program_id'); $table->index('participant_id'); }); } public function down(): void { Schema::dropIfExists('certificates'); } };