feat: split name into first_name + last_name across users, persons, and companies
Cross-cutting migration affecting the entire stack: - Database: 3 migrations splitting name columns with data migration - Models: first_name/last_name on User, Person; contact_first_name/contact_last_name on Company; backward-compatible name accessors - API: all resources return first_name, last_name, full_name; assignablePersons endpoint updated - Requests: validation rules updated for all person/user/company forms - Services: VolunteerRegistrationService, ShiftAssignmentService, InvitationService updated - Frontend: TypeScript types, Zod schemas, all forms split into Voornaam/Achternaam fields - Display: all person/user name references use full_name; initials use first_name[0]+last_name[0] - Tests: all 371 tests passing - Docs: SCHEMA.md and API.md updated Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -64,7 +64,9 @@ final class InvitationController extends Controller
|
||||
return $this->success([
|
||||
'user' => [
|
||||
'id' => $user->id,
|
||||
'name' => $user->name,
|
||||
'first_name' => $user->first_name,
|
||||
'last_name' => $user->last_name,
|
||||
'full_name' => $user->full_name,
|
||||
'email' => $user->email,
|
||||
],
|
||||
'token' => $sanctumToken,
|
||||
|
||||
@@ -145,7 +145,8 @@ final class ShiftAssignmentController extends Controller
|
||||
$persons = Person::where('event_id', $festivalEventId)
|
||||
->where('status', PersonStatus::APPROVED)
|
||||
->with('crowdType')
|
||||
->orderBy('name')
|
||||
->orderBy('first_name')
|
||||
->orderBy('last_name')
|
||||
->get();
|
||||
|
||||
// Batch: tags for all persons with user_id
|
||||
@@ -184,7 +185,9 @@ final class ShiftAssignmentController extends Controller
|
||||
|
||||
return [
|
||||
'id' => $person->id,
|
||||
'name' => $person->name,
|
||||
'first_name' => $person->first_name,
|
||||
'last_name' => $person->last_name,
|
||||
'full_name' => $person->full_name,
|
||||
'email' => $person->email,
|
||||
'status' => $person->status,
|
||||
'crowd_type' => $person->crowdType ? [
|
||||
@@ -222,7 +225,7 @@ final class ShiftAssignmentController extends Controller
|
||||
->sortBy([
|
||||
['already_assigned', 'asc'],
|
||||
['is_available', 'desc'],
|
||||
['name', 'asc'],
|
||||
['first_name', 'asc'],
|
||||
])
|
||||
->values();
|
||||
|
||||
|
||||
@@ -22,7 +22,8 @@ final class AcceptInvitationRequest extends FormRequest
|
||||
$userExists = $invitation && User::where('email', $invitation->email)->exists();
|
||||
|
||||
return [
|
||||
'name' => [$userExists ? 'nullable' : 'required', 'string', 'max:255'],
|
||||
'first_name' => [$userExists ? 'nullable' : 'required', 'string', 'max:255'],
|
||||
'last_name' => [$userExists ? 'nullable' : 'required', 'string', 'max:255'],
|
||||
'password' => [$userExists ? 'nullable' : 'required', 'string', 'min:8', 'confirmed'],
|
||||
];
|
||||
}
|
||||
|
||||
@@ -19,7 +19,8 @@ final class StoreCompanyRequest extends FormRequest
|
||||
return [
|
||||
'name' => ['required', 'string', 'max:100'],
|
||||
'type' => ['required', 'in:supplier,partner,agency,venue,other'],
|
||||
'contact_name' => ['nullable', 'string', 'max:100'],
|
||||
'contact_first_name' => ['nullable', 'string', 'max:100'],
|
||||
'contact_last_name' => ['nullable', 'string', 'max:100'],
|
||||
'contact_email' => ['nullable', 'email', 'max:100'],
|
||||
'contact_phone' => ['nullable', 'string', 'max:30'],
|
||||
];
|
||||
|
||||
@@ -18,7 +18,8 @@ final class StorePersonRequest extends FormRequest
|
||||
{
|
||||
return [
|
||||
'crowd_type_id' => ['required', 'ulid', 'exists:crowd_types,id'],
|
||||
'name' => ['required', 'string', 'max:255'],
|
||||
'first_name' => ['required', 'string', 'max:255'],
|
||||
'last_name' => ['required', 'string', 'max:255'],
|
||||
'email' => ['required', 'email', 'max:255'],
|
||||
'phone' => ['nullable', 'string', 'max:30'],
|
||||
'company_id' => ['nullable', 'ulid', 'exists:companies,id'],
|
||||
|
||||
@@ -19,7 +19,8 @@ final class UpdateCompanyRequest extends FormRequest
|
||||
return [
|
||||
'name' => ['sometimes', 'string', 'max:100'],
|
||||
'type' => ['sometimes', 'in:supplier,partner,agency,venue,other'],
|
||||
'contact_name' => ['nullable', 'string', 'max:100'],
|
||||
'contact_first_name' => ['nullable', 'string', 'max:100'],
|
||||
'contact_last_name' => ['nullable', 'string', 'max:100'],
|
||||
'contact_email' => ['nullable', 'email', 'max:100'],
|
||||
'contact_phone' => ['nullable', 'string', 'max:30'],
|
||||
];
|
||||
|
||||
@@ -18,7 +18,8 @@ final class UpdatePersonRequest extends FormRequest
|
||||
{
|
||||
return [
|
||||
'crowd_type_id' => ['sometimes', 'ulid', 'exists:crowd_types,id'],
|
||||
'name' => ['sometimes', 'string', 'max:255'],
|
||||
'first_name' => ['sometimes', 'string', 'max:255'],
|
||||
'last_name' => ['sometimes', 'string', 'max:255'],
|
||||
'email' => ['sometimes', 'email', 'max:255'],
|
||||
'phone' => ['nullable', 'string', 'max:30'],
|
||||
'company_id' => ['nullable', 'ulid', 'exists:companies,id'],
|
||||
|
||||
@@ -19,7 +19,8 @@ final class VolunteerRegistrationRequest extends FormRequest
|
||||
|
||||
if ($user) {
|
||||
$this->merge([
|
||||
'name' => $user->name,
|
||||
'first_name' => $user->first_name,
|
||||
'last_name' => $user->last_name,
|
||||
'email' => $user->email,
|
||||
'_authenticated' => true,
|
||||
]);
|
||||
@@ -30,7 +31,8 @@ final class VolunteerRegistrationRequest extends FormRequest
|
||||
public function rules(): array
|
||||
{
|
||||
return [
|
||||
'name' => ['required_without:_authenticated', 'string', 'max:255'],
|
||||
'first_name' => ['required_without:_authenticated', 'string', 'max:255'],
|
||||
'last_name' => ['required_without:_authenticated', 'string', 'max:255'],
|
||||
'email' => ['required_without:_authenticated', 'email', 'max:255'],
|
||||
'phone' => ['nullable', 'string', 'max:50'],
|
||||
|
||||
|
||||
@@ -16,7 +16,9 @@ final class CompanyResource extends JsonResource
|
||||
'organisation_id' => $this->organisation_id,
|
||||
'name' => $this->name,
|
||||
'type' => $this->type,
|
||||
'contact_name' => $this->contact_name,
|
||||
'contact_first_name' => $this->contact_first_name,
|
||||
'contact_last_name' => $this->contact_last_name,
|
||||
'contact_full_name' => $this->contact_full_name,
|
||||
'contact_email' => $this->contact_email,
|
||||
'contact_phone' => $this->contact_phone,
|
||||
'persons_count' => $this->whenCounted('persons'),
|
||||
|
||||
@@ -13,7 +13,9 @@ final class MeResource extends JsonResource
|
||||
{
|
||||
return [
|
||||
'id' => $this->id,
|
||||
'name' => $this->name,
|
||||
'first_name' => $this->first_name,
|
||||
'last_name' => $this->last_name,
|
||||
'full_name' => $this->full_name,
|
||||
'email' => $this->email,
|
||||
'timezone' => $this->timezone,
|
||||
'locale' => $this->locale,
|
||||
|
||||
@@ -13,7 +13,9 @@ final class MemberResource extends JsonResource
|
||||
{
|
||||
return [
|
||||
'id' => $this->id,
|
||||
'name' => $this->name,
|
||||
'first_name' => $this->first_name,
|
||||
'last_name' => $this->last_name,
|
||||
'full_name' => $this->full_name,
|
||||
'email' => $this->email,
|
||||
'role' => $this->pivot?->role,
|
||||
'avatar' => $this->avatar,
|
||||
|
||||
@@ -14,7 +14,9 @@ final class PersonResource extends JsonResource
|
||||
return [
|
||||
'id' => $this->id,
|
||||
'event_id' => $this->event_id,
|
||||
'name' => $this->name,
|
||||
'first_name' => $this->first_name,
|
||||
'last_name' => $this->last_name,
|
||||
'full_name' => $this->full_name,
|
||||
'email' => $this->email,
|
||||
'phone' => $this->phone,
|
||||
'status' => $this->status,
|
||||
@@ -33,7 +35,9 @@ final class PersonResource extends JsonResource
|
||||
'match_id' => $match->id,
|
||||
'matched_user' => [
|
||||
'id' => $match->matchedUser->id,
|
||||
'name' => $match->matchedUser->name,
|
||||
'first_name' => $match->matchedUser->first_name,
|
||||
'last_name' => $match->matchedUser->last_name,
|
||||
'full_name' => $match->matchedUser->full_name,
|
||||
'email' => $match->matchedUser->email,
|
||||
],
|
||||
'matched_on' => $match->matched_on->value,
|
||||
|
||||
@@ -13,7 +13,9 @@ final class UserResource extends JsonResource
|
||||
{
|
||||
return [
|
||||
'id' => $this->id,
|
||||
'name' => $this->name,
|
||||
'first_name' => $this->first_name,
|
||||
'last_name' => $this->last_name,
|
||||
'full_name' => $this->full_name,
|
||||
'email' => $this->email,
|
||||
'roles' => $this->getRoleNames()->values()->all(),
|
||||
'timezone' => $this->timezone,
|
||||
|
||||
@@ -22,11 +22,21 @@ final class Company extends Model
|
||||
'organisation_id',
|
||||
'name',
|
||||
'type',
|
||||
'contact_name',
|
||||
'contact_first_name',
|
||||
'contact_last_name',
|
||||
'contact_email',
|
||||
'contact_phone',
|
||||
];
|
||||
|
||||
public function getContactFullNameAttribute(): ?string
|
||||
{
|
||||
if (! $this->contact_first_name) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return trim("{$this->contact_first_name} {$this->contact_last_name}");
|
||||
}
|
||||
|
||||
public function organisation(): BelongsTo
|
||||
{
|
||||
return $this->belongsTo(Organisation::class);
|
||||
|
||||
@@ -28,7 +28,8 @@ final class Person extends Model
|
||||
'event_id',
|
||||
'crowd_type_id',
|
||||
'company_id',
|
||||
'name',
|
||||
'first_name',
|
||||
'last_name',
|
||||
'email',
|
||||
'phone',
|
||||
'status',
|
||||
@@ -37,6 +38,16 @@ final class Person extends Model
|
||||
'custom_fields',
|
||||
];
|
||||
|
||||
public function getFullNameAttribute(): string
|
||||
{
|
||||
return trim("{$this->first_name} {$this->last_name}");
|
||||
}
|
||||
|
||||
public function getNameAttribute(): string
|
||||
{
|
||||
return $this->full_name;
|
||||
}
|
||||
|
||||
protected function casts(): array
|
||||
{
|
||||
return [
|
||||
|
||||
@@ -24,7 +24,8 @@ final class User extends Authenticatable
|
||||
use SoftDeletes;
|
||||
|
||||
protected $fillable = [
|
||||
'name',
|
||||
'first_name',
|
||||
'last_name',
|
||||
'email',
|
||||
'password',
|
||||
'timezone',
|
||||
@@ -32,6 +33,16 @@ final class User extends Authenticatable
|
||||
'avatar',
|
||||
];
|
||||
|
||||
public function getFullNameAttribute(): string
|
||||
{
|
||||
return trim("{$this->first_name} {$this->last_name}");
|
||||
}
|
||||
|
||||
public function getNameAttribute(): string
|
||||
{
|
||||
return $this->full_name;
|
||||
}
|
||||
|
||||
protected $hidden = [
|
||||
'password',
|
||||
'remember_token',
|
||||
|
||||
@@ -76,7 +76,8 @@ final class InvitationService
|
||||
}
|
||||
|
||||
$user = User::create([
|
||||
'name' => Str::before($invitation->email, '@'),
|
||||
'first_name' => Str::before($invitation->email, '@'),
|
||||
'last_name' => '',
|
||||
'email' => $invitation->email,
|
||||
'password' => $password,
|
||||
'email_verified_at' => now(),
|
||||
|
||||
@@ -102,7 +102,7 @@ final class ShiftAssignmentService
|
||||
->performedOn($existing)
|
||||
->withProperties([
|
||||
'previous_status' => $previousStatus,
|
||||
'person_name' => $person->name,
|
||||
'person_name' => $person->full_name,
|
||||
])
|
||||
->log('shift_assignment.reactivated');
|
||||
|
||||
@@ -124,7 +124,7 @@ final class ShiftAssignmentService
|
||||
'filled_slots' => $filledSlots,
|
||||
'slots_total' => $shift->slots_total,
|
||||
'person_id' => $person->id,
|
||||
'person_name' => $person->name,
|
||||
'person_name' => $person->full_name,
|
||||
])
|
||||
->log('shift.overbooked_assignment');
|
||||
}
|
||||
|
||||
@@ -50,7 +50,8 @@ final class VolunteerRegistrationService
|
||||
[
|
||||
'user_id' => $user?->id,
|
||||
'crowd_type_id' => $volunteerCrowdType->id,
|
||||
'name' => $user?->name ?? $validated['name'],
|
||||
'first_name' => $user?->first_name ?? $validated['first_name'],
|
||||
'last_name' => $user?->last_name ?? $validated['last_name'],
|
||||
'phone' => $validated['phone'] ?? null,
|
||||
'status' => PersonStatus::PENDING,
|
||||
'custom_fields' => [
|
||||
@@ -119,7 +120,7 @@ final class VolunteerRegistrationService
|
||||
return;
|
||||
}
|
||||
|
||||
if ($existing->status !== PersonStatus::REJECTED) {
|
||||
if ($existing->status !== PersonStatus::REJECTED->value) {
|
||||
throw ValidationException::withMessages([
|
||||
'email' => ['Already registered for this event.'],
|
||||
]);
|
||||
|
||||
@@ -18,7 +18,8 @@ final class CompanyFactory extends Factory
|
||||
'organisation_id' => Organisation::factory(),
|
||||
'name' => fake('nl_NL')->company(),
|
||||
'type' => fake()->randomElement(['supplier', 'partner', 'agency', 'venue', 'other']),
|
||||
'contact_name' => fake('nl_NL')->name(),
|
||||
'contact_first_name' => fake('nl_NL')->firstName(),
|
||||
'contact_last_name' => fake('nl_NL')->lastName(),
|
||||
'contact_email' => fake()->companyEmail(),
|
||||
'contact_phone' => fake('nl_NL')->phoneNumber(),
|
||||
];
|
||||
|
||||
@@ -18,7 +18,8 @@ final class PersonFactory extends Factory
|
||||
return [
|
||||
'event_id' => Event::factory(),
|
||||
'crowd_type_id' => CrowdType::factory(),
|
||||
'name' => fake('nl_NL')->name(),
|
||||
'first_name' => fake('nl_NL')->firstName(),
|
||||
'last_name' => fake('nl_NL')->lastName(),
|
||||
'email' => fake()->unique()->safeEmail(),
|
||||
'phone' => fake('nl_NL')->phoneNumber(),
|
||||
'status' => 'pending',
|
||||
@@ -31,4 +32,9 @@ final class PersonFactory extends Factory
|
||||
{
|
||||
return $this->state(fn () => ['status' => 'approved']);
|
||||
}
|
||||
|
||||
public function rejected(): static
|
||||
{
|
||||
return $this->state(fn () => ['status' => 'rejected']);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -18,7 +18,8 @@ final class UserFactory extends Factory
|
||||
public function definition(): array
|
||||
{
|
||||
return [
|
||||
'name' => fake()->name(),
|
||||
'first_name' => fake('nl_NL')->firstName(),
|
||||
'last_name' => fake('nl_NL')->lastName(),
|
||||
'email' => fake()->unique()->safeEmail(),
|
||||
'email_verified_at' => now(),
|
||||
'password' => static::$password ??= Hash::make('password'),
|
||||
|
||||
@@ -0,0 +1,48 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
use Illuminate\Database\Migrations\Migration;
|
||||
use Illuminate\Database\Schema\Blueprint;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
use Illuminate\Support\Facades\Schema;
|
||||
|
||||
return new class extends Migration
|
||||
{
|
||||
public function up(): void
|
||||
{
|
||||
Schema::table('users', function (Blueprint $table) {
|
||||
$table->string('first_name')->after('id');
|
||||
$table->string('last_name')->after('first_name');
|
||||
});
|
||||
|
||||
DB::table('users')->get()->each(function ($user) {
|
||||
$parts = explode(' ', $user->name, 2);
|
||||
DB::table('users')->where('id', $user->id)->update([
|
||||
'first_name' => $parts[0],
|
||||
'last_name' => $parts[1] ?? '',
|
||||
]);
|
||||
});
|
||||
|
||||
Schema::table('users', function (Blueprint $table) {
|
||||
$table->dropColumn('name');
|
||||
});
|
||||
}
|
||||
|
||||
public function down(): void
|
||||
{
|
||||
Schema::table('users', function (Blueprint $table) {
|
||||
$table->string('name')->after('id');
|
||||
});
|
||||
|
||||
DB::table('users')->get()->each(function ($user) {
|
||||
DB::table('users')->where('id', $user->id)->update([
|
||||
'name' => trim("{$user->first_name} {$user->last_name}"),
|
||||
]);
|
||||
});
|
||||
|
||||
Schema::table('users', function (Blueprint $table) {
|
||||
$table->dropColumn(['first_name', 'last_name']);
|
||||
});
|
||||
}
|
||||
};
|
||||
@@ -0,0 +1,48 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
use Illuminate\Database\Migrations\Migration;
|
||||
use Illuminate\Database\Schema\Blueprint;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
use Illuminate\Support\Facades\Schema;
|
||||
|
||||
return new class extends Migration
|
||||
{
|
||||
public function up(): void
|
||||
{
|
||||
Schema::table('persons', function (Blueprint $table) {
|
||||
$table->string('first_name')->after('company_id');
|
||||
$table->string('last_name')->after('first_name');
|
||||
});
|
||||
|
||||
DB::table('persons')->get()->each(function ($person) {
|
||||
$parts = explode(' ', $person->name, 2);
|
||||
DB::table('persons')->where('id', $person->id)->update([
|
||||
'first_name' => $parts[0],
|
||||
'last_name' => $parts[1] ?? '',
|
||||
]);
|
||||
});
|
||||
|
||||
Schema::table('persons', function (Blueprint $table) {
|
||||
$table->dropColumn('name');
|
||||
});
|
||||
}
|
||||
|
||||
public function down(): void
|
||||
{
|
||||
Schema::table('persons', function (Blueprint $table) {
|
||||
$table->string('name')->after('company_id');
|
||||
});
|
||||
|
||||
DB::table('persons')->get()->each(function ($person) {
|
||||
DB::table('persons')->where('id', $person->id)->update([
|
||||
'name' => trim("{$person->first_name} {$person->last_name}"),
|
||||
]);
|
||||
});
|
||||
|
||||
Schema::table('persons', function (Blueprint $table) {
|
||||
$table->dropColumn(['first_name', 'last_name']);
|
||||
});
|
||||
}
|
||||
};
|
||||
@@ -0,0 +1,48 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
use Illuminate\Database\Migrations\Migration;
|
||||
use Illuminate\Database\Schema\Blueprint;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
use Illuminate\Support\Facades\Schema;
|
||||
|
||||
return new class extends Migration
|
||||
{
|
||||
public function up(): void
|
||||
{
|
||||
Schema::table('companies', function (Blueprint $table) {
|
||||
$table->string('contact_first_name')->nullable()->after('type');
|
||||
$table->string('contact_last_name')->nullable()->after('contact_first_name');
|
||||
});
|
||||
|
||||
DB::table('companies')->whereNotNull('contact_name')->get()->each(function ($company) {
|
||||
$parts = explode(' ', $company->contact_name, 2);
|
||||
DB::table('companies')->where('id', $company->id)->update([
|
||||
'contact_first_name' => $parts[0],
|
||||
'contact_last_name' => $parts[1] ?? '',
|
||||
]);
|
||||
});
|
||||
|
||||
Schema::table('companies', function (Blueprint $table) {
|
||||
$table->dropColumn('contact_name');
|
||||
});
|
||||
}
|
||||
|
||||
public function down(): void
|
||||
{
|
||||
Schema::table('companies', function (Blueprint $table) {
|
||||
$table->string('contact_name')->nullable()->after('type');
|
||||
});
|
||||
|
||||
DB::table('companies')->whereNotNull('contact_first_name')->get()->each(function ($company) {
|
||||
DB::table('companies')->where('id', $company->id)->update([
|
||||
'contact_name' => trim("{$company->contact_first_name} {$company->contact_last_name}"),
|
||||
]);
|
||||
});
|
||||
|
||||
Schema::table('companies', function (Blueprint $table) {
|
||||
$table->dropColumn(['contact_first_name', 'contact_last_name']);
|
||||
});
|
||||
}
|
||||
};
|
||||
@@ -16,7 +16,8 @@ class DatabaseSeeder extends Seeder
|
||||
$this->call(RoleSeeder::class);
|
||||
|
||||
$admin = User::create([
|
||||
'name' => 'Super Admin',
|
||||
'first_name' => 'Super',
|
||||
'last_name' => 'Admin',
|
||||
'email' => 'admin@crewli.app',
|
||||
'password' => Hash::make('password'),
|
||||
]);
|
||||
|
||||
@@ -71,19 +71,20 @@ class DevSeeder extends Seeder
|
||||
// ── Users (8) ──
|
||||
|
||||
$usersData = [
|
||||
['email' => 'admin@crewli.test', 'name' => 'Super Admin', 'app_role' => 'super_admin', 'org_role' => 'org_admin'],
|
||||
['email' => 'bert@feestfabriek.nl', 'name' => 'Bert Hausmans', 'org_role' => 'org_admin'],
|
||||
['email' => 'lisa@feestfabriek.nl', 'name' => 'Lisa van den Berg', 'org_role' => 'org_member'],
|
||||
['email' => 'ahmed@feestfabriek.nl', 'name' => 'Ahmed Yilmaz', 'org_role' => 'org_member'],
|
||||
['email' => 'sara@feestfabriek.nl', 'name' => 'Sara de Groot', 'org_role' => 'org_member'],
|
||||
['email' => 'tom@feestfabriek.nl', 'name' => 'Tom Visser', 'org_role' => 'org_member'],
|
||||
['email' => 'nina@feestfabriek.nl', 'name' => 'Nina Jansen', 'org_role' => 'org_member'],
|
||||
['email' => 'mark@feestfabriek.nl', 'name' => 'Mark de Boer', 'org_role' => 'org_member'],
|
||||
['email' => 'admin@crewli.test', 'first_name' => 'Super', 'last_name' => 'Admin', 'app_role' => 'super_admin', 'org_role' => 'org_admin'],
|
||||
['email' => 'bert@feestfabriek.nl', 'first_name' => 'Bert', 'last_name' => 'Hausmans', 'org_role' => 'org_admin'],
|
||||
['email' => 'lisa@feestfabriek.nl', 'first_name' => 'Lisa', 'last_name' => 'van den Berg', 'org_role' => 'org_member'],
|
||||
['email' => 'ahmed@feestfabriek.nl', 'first_name' => 'Ahmed', 'last_name' => 'Yilmaz', 'org_role' => 'org_member'],
|
||||
['email' => 'sara@feestfabriek.nl', 'first_name' => 'Sara', 'last_name' => 'de Groot', 'org_role' => 'org_member'],
|
||||
['email' => 'tom@feestfabriek.nl', 'first_name' => 'Tom', 'last_name' => 'Visser', 'org_role' => 'org_member'],
|
||||
['email' => 'nina@feestfabriek.nl', 'first_name' => 'Nina', 'last_name' => 'Jansen', 'org_role' => 'org_member'],
|
||||
['email' => 'mark@feestfabriek.nl', 'first_name' => 'Mark', 'last_name' => 'de Boer', 'org_role' => 'org_member'],
|
||||
];
|
||||
|
||||
foreach ($usersData as $data) {
|
||||
$user = User::create([
|
||||
'name' => $data['name'],
|
||||
'first_name' => $data['first_name'],
|
||||
'last_name' => $data['last_name'],
|
||||
'email' => $data['email'],
|
||||
'password' => Hash::make('password'),
|
||||
]);
|
||||
@@ -99,12 +100,12 @@ class DevSeeder extends Seeder
|
||||
// ── Companies (6) ──
|
||||
|
||||
$companiesData = [
|
||||
['name' => 'Tap & Co Horeca', 'type' => 'supplier', 'contact_name' => 'Jan Tapper', 'contact_email' => 'jan@tapco.nl', 'contact_phone' => '+31612340001'],
|
||||
['name' => 'SecureEvent BV', 'type' => 'supplier', 'contact_name' => 'Klaas Veilig', 'contact_email' => 'klaas@secureevent.nl', 'contact_phone' => '+31612340002'],
|
||||
['name' => 'Podiumtechniek Rijnmond', 'type' => 'supplier', 'contact_name' => 'Pieter Geluid', 'contact_email' => 'pieter@podiumtechniek.nl', 'contact_phone' => '+31612340003'],
|
||||
['name' => 'Brouwerij De Schelde', 'type' => 'partner', 'contact_name' => 'Eva Brouwer', 'contact_email' => 'eva@brouwerijdeschelde.nl', 'contact_phone' => '+31612340004'],
|
||||
['name' => 'Van Dijk Catering', 'type' => 'supplier', 'contact_name' => 'Maria van Dijk', 'contact_email' => 'maria@vandijkcatering.nl', 'contact_phone' => '+31612340005'],
|
||||
['name' => 'Rotterdam Festivals', 'type' => 'agency', 'contact_name' => 'Femke de Wit', 'contact_email' => 'femke@rotterdamfestivals.nl', 'contact_phone' => '+31612340006'],
|
||||
['name' => 'Tap & Co Horeca', 'type' => 'supplier', 'contact_first_name' => 'Jan', 'contact_last_name' => 'Tapper', 'contact_email' => 'jan@tapco.nl', 'contact_phone' => '+31612340001'],
|
||||
['name' => 'SecureEvent BV', 'type' => 'supplier', 'contact_first_name' => 'Klaas', 'contact_last_name' => 'Veilig', 'contact_email' => 'klaas@secureevent.nl', 'contact_phone' => '+31612340002'],
|
||||
['name' => 'Podiumtechniek Rijnmond', 'type' => 'supplier', 'contact_first_name' => 'Pieter', 'contact_last_name' => 'Geluid', 'contact_email' => 'pieter@podiumtechniek.nl', 'contact_phone' => '+31612340003'],
|
||||
['name' => 'Brouwerij De Schelde', 'type' => 'partner', 'contact_first_name' => 'Eva', 'contact_last_name' => 'Brouwer', 'contact_email' => 'eva@brouwerijdeschelde.nl', 'contact_phone' => '+31612340004'],
|
||||
['name' => 'Van Dijk Catering', 'type' => 'supplier', 'contact_first_name' => 'Maria', 'contact_last_name' => 'van Dijk', 'contact_email' => 'maria@vandijkcatering.nl', 'contact_phone' => '+31612340005'],
|
||||
['name' => 'Rotterdam Festivals', 'type' => 'agency', 'contact_first_name' => 'Femke', 'contact_last_name' => 'de Wit', 'contact_email' => 'femke@rotterdamfestivals.nl', 'contact_phone' => '+31612340006'],
|
||||
];
|
||||
|
||||
foreach ($companiesData as $data) {
|
||||
@@ -455,58 +456,59 @@ class DevSeeder extends Seeder
|
||||
// Create user accounts for volunteers who need user_id links
|
||||
$volunteerUsers = [];
|
||||
foreach ([
|
||||
['name' => 'Jan de Vries', 'email' => 'jan@gmail.com'],
|
||||
['name' => 'Ahmed Hassan', 'email' => 'ahmed.h@gmail.com'],
|
||||
['name' => 'Tom Visser', 'email' => 'tom.visser@gmail.com'],
|
||||
['name' => 'Lotte de Jong', 'email' => 'lotte@gmail.com'],
|
||||
['name' => 'Pieter Geluid', 'email' => 'pieter@podiumtechniek.nl'],
|
||||
['first_name' => 'Jan', 'last_name' => 'de Vries', 'email' => 'jan@gmail.com'],
|
||||
['first_name' => 'Ahmed', 'last_name' => 'Hassan', 'email' => 'ahmed.h@gmail.com'],
|
||||
['first_name' => 'Tom', 'last_name' => 'Visser', 'email' => 'tom.visser@gmail.com'],
|
||||
['first_name' => 'Lotte', 'last_name' => 'de Jong', 'email' => 'lotte@gmail.com'],
|
||||
['first_name' => 'Pieter', 'last_name' => 'Geluid', 'email' => 'pieter@podiumtechniek.nl'],
|
||||
] as $u) {
|
||||
$volunteerUsers[$u['email']] = User::create([
|
||||
'name' => $u['name'],
|
||||
'first_name' => $u['first_name'],
|
||||
'last_name' => $u['last_name'],
|
||||
'email' => $u['email'],
|
||||
'password' => Hash::make('password'),
|
||||
]);
|
||||
}
|
||||
|
||||
// 15 named volunteers
|
||||
$jan = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'user_id' => $volunteerUsers['jan@gmail.com']->id, 'name' => 'Jan de Vries', 'email' => 'jan@gmail.com', 'phone' => '+31612345001', 'status' => 'approved']);
|
||||
$lisaB = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'name' => 'Lisa Bakker', 'email' => 'lisa.bakker@hotmail.com', 'phone' => '+31612345002', 'status' => 'approved']);
|
||||
$ahmedP = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'user_id' => $volunteerUsers['ahmed.h@gmail.com']->id, 'name' => 'Ahmed Hassan', 'email' => 'ahmed.h@gmail.com', 'phone' => '+31612345003', 'status' => 'approved']);
|
||||
$saraJ = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'name' => 'Sara Jansen', 'email' => 'sara.j@outlook.com', 'phone' => '+31612345004', 'status' => 'approved']);
|
||||
$tomV = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'user_id' => $volunteerUsers['tom.visser@gmail.com']->id, 'name' => 'Tom Visser', 'email' => 'tom.visser@gmail.com', 'phone' => '+31612345005', 'status' => 'approved']);
|
||||
$fatima = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'name' => 'Fatima El Amrani', 'email' => 'fatima@gmail.com', 'phone' => '+31612345006', 'status' => 'approved']);
|
||||
$daan = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'name' => 'Daan Smit', 'email' => 'daan.smit@gmail.com', 'phone' => '+31612345007', 'status' => 'pending']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'name' => 'Sophie Mulder', 'email' => 'sophie.m@hotmail.com', 'phone' => '+31612345008', 'status' => 'pending']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'name' => 'Jesse van Dijk', 'email' => 'jesse@gmail.com', 'phone' => '+31612345009', 'status' => 'applied']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'name' => 'Noa Hendriks', 'email' => 'noa.h@outlook.com', 'phone' => '+31612345010', 'status' => 'applied']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'name' => 'Kevin Bos', 'email' => 'kevin.bos@gmail.com', 'phone' => '+31612345011', 'status' => 'rejected', 'admin_notes' => 'Vorig jaar no-show']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'name' => 'Priya Sharma', 'email' => 'priya@gmail.com', 'phone' => '+31612345012', 'status' => 'invited']);
|
||||
$lotte = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'user_id' => $volunteerUsers['lotte@gmail.com']->id, 'name' => 'Lotte de Jong', 'email' => 'lotte@gmail.com', 'phone' => '+31612345013', 'status' => 'approved']);
|
||||
$robin = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'name' => 'Robin Peters', 'email' => 'robin.p@hotmail.com', 'phone' => '+31612345014', 'status' => 'approved']);
|
||||
$emma = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'name' => 'Emma Willems', 'email' => 'emma.w@gmail.com', 'phone' => '+31612345015', 'status' => 'no_show', 'is_blacklisted' => true]);
|
||||
$jan = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'user_id' => $volunteerUsers['jan@gmail.com']->id, 'first_name' => 'Jan', 'last_name' => 'de Vries', 'email' => 'jan@gmail.com', 'phone' => '+31612345001', 'status' => 'approved']);
|
||||
$lisaB = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'first_name' => 'Lisa', 'last_name' => 'Bakker', 'email' => 'lisa.bakker@hotmail.com', 'phone' => '+31612345002', 'status' => 'approved']);
|
||||
$ahmedP = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'user_id' => $volunteerUsers['ahmed.h@gmail.com']->id, 'first_name' => 'Ahmed', 'last_name' => 'Hassan', 'email' => 'ahmed.h@gmail.com', 'phone' => '+31612345003', 'status' => 'approved']);
|
||||
$saraJ = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'first_name' => 'Sara', 'last_name' => 'Jansen', 'email' => 'sara.j@outlook.com', 'phone' => '+31612345004', 'status' => 'approved']);
|
||||
$tomV = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'user_id' => $volunteerUsers['tom.visser@gmail.com']->id, 'first_name' => 'Tom', 'last_name' => 'Visser', 'email' => 'tom.visser@gmail.com', 'phone' => '+31612345005', 'status' => 'approved']);
|
||||
$fatima = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'first_name' => 'Fatima', 'last_name' => 'El Amrani', 'email' => 'fatima@gmail.com', 'phone' => '+31612345006', 'status' => 'approved']);
|
||||
$daan = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'first_name' => 'Daan', 'last_name' => 'Smit', 'email' => 'daan.smit@gmail.com', 'phone' => '+31612345007', 'status' => 'pending']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'first_name' => 'Sophie', 'last_name' => 'Mulder', 'email' => 'sophie.m@hotmail.com', 'phone' => '+31612345008', 'status' => 'pending']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'first_name' => 'Jesse', 'last_name' => 'van Dijk', 'email' => 'jesse@gmail.com', 'phone' => '+31612345009', 'status' => 'applied']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'first_name' => 'Noa', 'last_name' => 'Hendriks', 'email' => 'noa.h@outlook.com', 'phone' => '+31612345010', 'status' => 'applied']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'first_name' => 'Kevin', 'last_name' => 'Bos', 'email' => 'kevin.bos@gmail.com', 'phone' => '+31612345011', 'status' => 'rejected', 'admin_notes' => 'Vorig jaar no-show']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'first_name' => 'Priya', 'last_name' => 'Sharma', 'email' => 'priya@gmail.com', 'phone' => '+31612345012', 'status' => 'invited']);
|
||||
$lotte = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'user_id' => $volunteerUsers['lotte@gmail.com']->id, 'first_name' => 'Lotte', 'last_name' => 'de Jong', 'email' => 'lotte@gmail.com', 'phone' => '+31612345013', 'status' => 'approved']);
|
||||
$robin = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'first_name' => 'Robin', 'last_name' => 'Peters', 'email' => 'robin.p@hotmail.com', 'phone' => '+31612345014', 'status' => 'approved']);
|
||||
$emma = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $vol, 'first_name' => 'Emma', 'last_name' => 'Willems', 'email' => 'emma.w@gmail.com', 'phone' => '+31612345015', 'status' => 'no_show', 'is_blacklisted' => true]);
|
||||
|
||||
// 6 named crew
|
||||
$klaas = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $crew, 'company_id' => $this->companies['SecureEvent BV']->id, 'name' => 'Klaas Veilig Jr.', 'email' => 'klaas.jr@secureevent.nl', 'phone' => '+31612345016', 'status' => 'approved']);
|
||||
$dennis = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $crew, 'company_id' => $this->companies['SecureEvent BV']->id, 'name' => 'Dennis Schild', 'email' => 'dennis@secureevent.nl', 'phone' => '+31612345017', 'status' => 'approved']);
|
||||
$pieter = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $crew, 'company_id' => $this->companies['Podiumtechniek Rijnmond']->id, 'user_id' => $volunteerUsers['pieter@podiumtechniek.nl']->id, 'name' => 'Pieter Geluid', 'email' => 'pieter@podiumtechniek.nl', 'phone' => '+31612345018', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $crew, 'company_id' => $this->companies['Podiumtechniek Rijnmond']->id, 'name' => 'Marco Licht', 'email' => 'marco@podiumtechniek.nl', 'phone' => '+31612345019', 'status' => 'approved']);
|
||||
$eva = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $crew, 'company_id' => $this->companies['Brouwerij De Schelde']->id, 'name' => 'Eva Brouwer', 'email' => 'eva@brouwerijdeschelde.nl', 'phone' => '+31612345020', 'status' => 'approved']);
|
||||
$maria = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $crew, 'company_id' => $this->companies['Van Dijk Catering']->id, 'name' => 'Maria van Dijk', 'email' => 'maria@vandijkcatering.nl', 'phone' => '+31612345021', 'status' => 'approved']);
|
||||
$klaas = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $crew, 'company_id' => $this->companies['SecureEvent BV']->id, 'first_name' => 'Klaas', 'last_name' => 'Veilig Jr.', 'email' => 'klaas.jr@secureevent.nl', 'phone' => '+31612345016', 'status' => 'approved']);
|
||||
$dennis = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $crew, 'company_id' => $this->companies['SecureEvent BV']->id, 'first_name' => 'Dennis', 'last_name' => 'Schild', 'email' => 'dennis@secureevent.nl', 'phone' => '+31612345017', 'status' => 'approved']);
|
||||
$pieter = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $crew, 'company_id' => $this->companies['Podiumtechniek Rijnmond']->id, 'user_id' => $volunteerUsers['pieter@podiumtechniek.nl']->id, 'first_name' => 'Pieter', 'last_name' => 'Geluid', 'email' => 'pieter@podiumtechniek.nl', 'phone' => '+31612345018', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $crew, 'company_id' => $this->companies['Podiumtechniek Rijnmond']->id, 'first_name' => 'Marco', 'last_name' => 'Licht', 'email' => 'marco@podiumtechniek.nl', 'phone' => '+31612345019', 'status' => 'approved']);
|
||||
$eva = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $crew, 'company_id' => $this->companies['Brouwerij De Schelde']->id, 'first_name' => 'Eva', 'last_name' => 'Brouwer', 'email' => 'eva@brouwerijdeschelde.nl', 'phone' => '+31612345020', 'status' => 'approved']);
|
||||
$maria = Person::create(['event_id' => $festival->id, 'crowd_type_id' => $crew, 'company_id' => $this->companies['Van Dijk Catering']->id, 'first_name' => 'Maria', 'last_name' => 'van Dijk', 'email' => 'maria@vandijkcatering.nl', 'phone' => '+31612345021', 'status' => 'approved']);
|
||||
|
||||
// 3 named press
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $press, 'name' => 'Joris van Laar', 'email' => 'joris@pers.nl', 'phone' => '+31612345022', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $press, 'name' => 'Tamara Smeets', 'email' => 'tamara@pers.nl', 'phone' => '+31612345023', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $press, 'name' => 'Sven Pieterse', 'email' => 'sven@pers.nl', 'phone' => '+31612345024', 'status' => 'pending']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $press, 'first_name' => 'Joris', 'last_name' => 'van Laar', 'email' => 'joris@pers.nl', 'phone' => '+31612345022', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $press, 'first_name' => 'Tamara', 'last_name' => 'Smeets', 'email' => 'tamara@pers.nl', 'phone' => '+31612345023', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $press, 'first_name' => 'Sven', 'last_name' => 'Pieterse', 'email' => 'sven@pers.nl', 'phone' => '+31612345024', 'status' => 'pending']);
|
||||
|
||||
// 4 named guests
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $guest, 'name' => 'Burgemeester Jan Slagter', 'email' => 'burgemeester@echt.nl', 'phone' => '+31612345025', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $guest, 'name' => 'Wethouder Petra Kamps', 'email' => 'wethouder@echt.nl', 'phone' => '+31612345026', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $guest, 'name' => 'Jan Sponsor', 'email' => 'sponsor@bedrijf.nl', 'phone' => '+31612345027', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $guest, 'name' => 'Gast van Artiest', 'email' => 'artiestgast@gmail.com', 'phone' => '+31612345028', 'status' => 'invited']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $guest, 'first_name' => 'Jan', 'last_name' => 'Slagter', 'email' => 'burgemeester@echt.nl', 'phone' => '+31612345025', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $guest, 'first_name' => 'Petra', 'last_name' => 'Kamps', 'email' => 'wethouder@echt.nl', 'phone' => '+31612345026', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $guest, 'first_name' => 'Jan', 'last_name' => 'Sponsor', 'email' => 'sponsor@bedrijf.nl', 'phone' => '+31612345027', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $guest, 'first_name' => 'Gast', 'last_name' => 'van Artiest', 'email' => 'artiestgast@gmail.com', 'phone' => '+31612345028', 'status' => 'invited']);
|
||||
|
||||
// 2 named suppliers
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $supplier, 'company_id' => $this->companies['Tap & Co Horeca']->id, 'name' => 'Hans Tapper', 'email' => 'hans@tapco.nl', 'phone' => '+31612345029', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $supplier, 'company_id' => $this->companies['Van Dijk Catering']->id, 'name' => 'Frank van Dijk', 'email' => 'frank@vandijkcatering.nl', 'phone' => '+31612345030', 'status' => 'pending']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $supplier, 'company_id' => $this->companies['Tap & Co Horeca']->id, 'first_name' => 'Hans', 'last_name' => 'Tapper', 'email' => 'hans@tapco.nl', 'phone' => '+31612345029', 'status' => 'approved']);
|
||||
Person::create(['event_id' => $festival->id, 'crowd_type_id' => $supplier, 'company_id' => $this->companies['Van Dijk Catering']->id, 'first_name' => 'Frank', 'last_name' => 'van Dijk', 'email' => 'frank@vandijkcatering.nl', 'phone' => '+31612345030', 'status' => 'pending']);
|
||||
|
||||
// ── Factory persons (120) ──
|
||||
|
||||
|
||||
@@ -199,8 +199,8 @@ class AssignablePersonsTest extends TestCase
|
||||
$shift1 = $this->createOpenShift();
|
||||
$shift2 = $this->createOpenShift(['festival_section_id' => $this->otherSection->id]);
|
||||
|
||||
$available = $this->createPerson(['name' => 'Anna Bakker']);
|
||||
$conflicted = $this->createPerson(['name' => 'Bob Jansen']);
|
||||
$available = $this->createPerson(['first_name' => 'Anna', 'last_name' => 'Bakker']);
|
||||
$conflicted = $this->createPerson(['first_name' => 'Bob', 'last_name' => 'Jansen']);
|
||||
|
||||
ShiftAssignment::factory()->create([
|
||||
'shift_id' => $shift1->id,
|
||||
|
||||
@@ -214,7 +214,8 @@ class RegistrationSettingsTest extends TestCase
|
||||
]);
|
||||
|
||||
$response = $this->postJson("/api/v1/events/{$event->id}/volunteer-register", [
|
||||
'name' => 'Test Vrijwilliger',
|
||||
'first_name' => 'Test',
|
||||
'last_name' => 'Vrijwilliger',
|
||||
'email' => 'test-section-pref@example.nl',
|
||||
'section_preferences' => [
|
||||
['section_name' => 'Backstage', 'priority' => 1],
|
||||
|
||||
@@ -54,7 +54,8 @@ class VolunteerRegistrationTest extends TestCase
|
||||
public function test_volunteer_can_register_with_all_fields(): void
|
||||
{
|
||||
$response = $this->postJson("/api/v1/events/{$this->event->id}/volunteer-register", [
|
||||
'name' => 'Jan de Vries',
|
||||
'first_name' => 'Jan',
|
||||
'last_name' => 'de Vries',
|
||||
'email' => 'jan@voorbeeld.nl',
|
||||
'phone' => '+31612345678',
|
||||
'tshirt_size' => 'L',
|
||||
@@ -76,7 +77,8 @@ class VolunteerRegistrationTest extends TestCase
|
||||
public function test_volunteer_can_register_with_minimal_fields(): void
|
||||
{
|
||||
$response = $this->postJson("/api/v1/events/{$this->event->id}/volunteer-register", [
|
||||
'name' => 'Sophie Bakker',
|
||||
'first_name' => 'Sophie',
|
||||
'last_name' => 'Bakker',
|
||||
'email' => 'sophie@voorbeeld.nl',
|
||||
]);
|
||||
|
||||
@@ -84,7 +86,8 @@ class VolunteerRegistrationTest extends TestCase
|
||||
|
||||
$this->assertDatabaseHas('persons', [
|
||||
'email' => 'sophie@voorbeeld.nl',
|
||||
'name' => 'Sophie Bakker',
|
||||
'first_name' => 'Sophie',
|
||||
'last_name' => 'Bakker',
|
||||
'event_id' => $this->event->id,
|
||||
]);
|
||||
}
|
||||
@@ -101,7 +104,8 @@ class VolunteerRegistrationTest extends TestCase
|
||||
TimeSlot::factory()->create(['event_id' => $festival->id]);
|
||||
|
||||
$response = $this->postJson("/api/v1/events/{$subEvent->id}/volunteer-register", [
|
||||
'name' => 'Pieter Jansen',
|
||||
'first_name' => 'Pieter',
|
||||
'last_name' => 'Jansen',
|
||||
'email' => 'pieter@voorbeeld.nl',
|
||||
]);
|
||||
|
||||
@@ -118,7 +122,8 @@ class VolunteerRegistrationTest extends TestCase
|
||||
$timeSlot2 = TimeSlot::factory()->create(['event_id' => $this->event->id]);
|
||||
|
||||
$response = $this->postJson("/api/v1/events/{$this->event->id}/volunteer-register", [
|
||||
'name' => 'Fleur Vermeer',
|
||||
'first_name' => 'Fleur',
|
||||
'last_name' => 'Vermeer',
|
||||
'email' => 'fleur@voorbeeld.nl',
|
||||
'availabilities' => [
|
||||
['time_slot_id' => $this->timeSlot->id, 'preference_level' => 4],
|
||||
@@ -145,7 +150,8 @@ class VolunteerRegistrationTest extends TestCase
|
||||
public function test_registration_stores_custom_fields(): void
|
||||
{
|
||||
$response = $this->postJson("/api/v1/events/{$this->event->id}/volunteer-register", [
|
||||
'name' => 'Daan Mulder',
|
||||
'first_name' => 'Daan',
|
||||
'last_name' => 'Mulder',
|
||||
'email' => 'daan@voorbeeld.nl',
|
||||
'tshirt_size' => 'XL',
|
||||
'motivation' => 'Ik vind festivals geweldig.',
|
||||
@@ -167,12 +173,14 @@ class VolunteerRegistrationTest extends TestCase
|
||||
public function test_duplicate_email_rejected(): void
|
||||
{
|
||||
$this->postJson("/api/v1/events/{$this->event->id}/volunteer-register", [
|
||||
'name' => 'Anna Smit',
|
||||
'first_name' => 'Anna',
|
||||
'last_name' => 'Smit',
|
||||
'email' => 'anna@voorbeeld.nl',
|
||||
]);
|
||||
|
||||
$response = $this->postJson("/api/v1/events/{$this->event->id}/volunteer-register", [
|
||||
'name' => 'Anna Smit',
|
||||
'first_name' => 'Anna',
|
||||
'last_name' => 'Smit',
|
||||
'email' => 'anna@voorbeeld.nl',
|
||||
]);
|
||||
|
||||
@@ -189,7 +197,8 @@ class VolunteerRegistrationTest extends TestCase
|
||||
]);
|
||||
|
||||
$response = $this->postJson("/api/v1/events/{$this->event->id}/volunteer-register", [
|
||||
'name' => 'Herkan Poging',
|
||||
'first_name' => 'Herkan',
|
||||
'last_name' => 'Poging',
|
||||
'email' => 'herkan@voorbeeld.nl',
|
||||
]);
|
||||
|
||||
@@ -209,7 +218,8 @@ class VolunteerRegistrationTest extends TestCase
|
||||
]);
|
||||
|
||||
$response = $this->postJson("/api/v1/events/{$draftEvent->id}/volunteer-register", [
|
||||
'name' => 'Test Persoon',
|
||||
'first_name' => 'Test',
|
||||
'last_name' => 'Persoon',
|
||||
'email' => 'test@voorbeeld.nl',
|
||||
]);
|
||||
|
||||
@@ -219,7 +229,8 @@ class VolunteerRegistrationTest extends TestCase
|
||||
public function test_invalid_time_slot_rejected(): void
|
||||
{
|
||||
$response = $this->postJson("/api/v1/events/{$this->event->id}/volunteer-register", [
|
||||
'name' => 'Bas van Dijk',
|
||||
'first_name' => 'Bas',
|
||||
'last_name' => 'van Dijk',
|
||||
'email' => 'bas@voorbeeld.nl',
|
||||
'availabilities' => [
|
||||
['time_slot_id' => '01JNONEXISTENT00000000000', 'preference_level' => 3],
|
||||
@@ -235,7 +246,8 @@ class VolunteerRegistrationTest extends TestCase
|
||||
public function test_authenticated_user_registration(): void
|
||||
{
|
||||
$user = User::factory()->create([
|
||||
'name' => 'Lisa de Groot',
|
||||
'first_name' => 'Lisa',
|
||||
'last_name' => 'de Groot',
|
||||
'email' => 'lisa@voorbeeld.nl',
|
||||
]);
|
||||
$this->organisation->users()->attach($user, ['role' => 'org_member']);
|
||||
@@ -255,7 +267,8 @@ class VolunteerRegistrationTest extends TestCase
|
||||
public function test_authenticated_ignores_request_email(): void
|
||||
{
|
||||
$user = User::factory()->create([
|
||||
'name' => 'Mark Visser',
|
||||
'first_name' => 'Mark',
|
||||
'last_name' => 'Visser',
|
||||
'email' => 'mark@voorbeeld.nl',
|
||||
]);
|
||||
$this->organisation->users()->attach($user, ['role' => 'org_member']);
|
||||
@@ -274,7 +287,8 @@ class VolunteerRegistrationTest extends TestCase
|
||||
public function test_authenticated_duplicate_rejected(): void
|
||||
{
|
||||
$user = User::factory()->create([
|
||||
'name' => 'Eva Hendriks',
|
||||
'first_name' => 'Eva',
|
||||
'last_name' => 'Hendriks',
|
||||
'email' => 'eva@voorbeeld.nl',
|
||||
]);
|
||||
$this->organisation->users()->attach($user, ['role' => 'org_member']);
|
||||
@@ -326,7 +340,7 @@ class VolunteerRegistrationTest extends TestCase
|
||||
|
||||
public function test_authenticated_user_gets_person(): void
|
||||
{
|
||||
$user = User::factory()->create(['name' => 'Karin Bos']);
|
||||
$user = User::factory()->create(['first_name' => 'Karin', 'last_name' => 'Bos']);
|
||||
$this->organisation->users()->attach($user, ['role' => 'org_member']);
|
||||
|
||||
Person::factory()->create([
|
||||
@@ -346,7 +360,7 @@ class VolunteerRegistrationTest extends TestCase
|
||||
|
||||
public function test_authenticated_user_no_person_returns_404(): void
|
||||
{
|
||||
$user = User::factory()->create(['name' => 'Tom Kuiper']);
|
||||
$user = User::factory()->create(['first_name' => 'Tom', 'last_name' => 'Kuiper']);
|
||||
Sanctum::actingAs($user);
|
||||
|
||||
$response = $this->getJson("/api/v1/portal/me?event_id={$this->event->id}");
|
||||
@@ -357,7 +371,7 @@ class VolunteerRegistrationTest extends TestCase
|
||||
|
||||
public function test_missing_event_id_returns_422(): void
|
||||
{
|
||||
$user = User::factory()->create(['name' => 'Sanne Bruin']);
|
||||
$user = User::factory()->create(['first_name' => 'Sanne', 'last_name' => 'Bruin']);
|
||||
Sanctum::actingAs($user);
|
||||
|
||||
$response = $this->getJson('/api/v1/portal/me');
|
||||
|
||||
@@ -24,7 +24,7 @@ class LoginTest extends TestCase
|
||||
$response->assertOk()
|
||||
->assertJsonStructure([
|
||||
'success',
|
||||
'data' => ['user' => ['id', 'name', 'email'], 'token'],
|
||||
'data' => ['user' => ['id', 'first_name', 'last_name', 'full_name', 'email'], 'token'],
|
||||
'message',
|
||||
])
|
||||
->assertJson(['success' => true]);
|
||||
|
||||
@@ -35,7 +35,7 @@ class MeTest extends TestCase
|
||||
->assertJsonStructure([
|
||||
'success',
|
||||
'data' => [
|
||||
'id', 'name', 'email', 'timezone', 'locale',
|
||||
'id', 'first_name', 'last_name', 'full_name', 'email', 'timezone', 'locale',
|
||||
'organisations', 'app_roles', 'permissions',
|
||||
],
|
||||
]);
|
||||
|
||||
@@ -100,7 +100,8 @@ class CompanyTest extends TestCase
|
||||
$response = $this->postJson("/api/v1/organisations/{$this->organisation->id}/companies", [
|
||||
'name' => 'Catering Janssen',
|
||||
'type' => 'supplier',
|
||||
'contact_name' => 'Jan Janssen',
|
||||
'contact_first_name' => 'Jan',
|
||||
'contact_last_name' => 'Janssen',
|
||||
'contact_email' => 'jan@janssen.nl',
|
||||
'contact_phone' => '+31612345678',
|
||||
]);
|
||||
|
||||
@@ -201,14 +201,16 @@ class FestivalEventTest extends TestCase
|
||||
Person::factory()->create([
|
||||
'event_id' => $this->festival->id,
|
||||
'crowd_type_id' => $this->crowdType->id,
|
||||
'name' => 'Jan Festivalmedewerker',
|
||||
'first_name' => 'Jan',
|
||||
'last_name' => 'Festivalmedewerker',
|
||||
]);
|
||||
|
||||
// Create a person on the sub-event
|
||||
Person::factory()->create([
|
||||
'event_id' => $this->subEvent->id,
|
||||
'crowd_type_id' => $this->crowdType->id,
|
||||
'name' => 'Piet Dagvrijwilliger',
|
||||
'first_name' => 'Piet',
|
||||
'last_name' => 'Dagvrijwilliger',
|
||||
]);
|
||||
|
||||
Sanctum::actingAs($this->orgAdmin);
|
||||
@@ -218,7 +220,7 @@ class FestivalEventTest extends TestCase
|
||||
|
||||
$response->assertOk();
|
||||
|
||||
$personNames = collect($response->json('data'))->pluck('name')->all();
|
||||
$personNames = collect($response->json('data'))->pluck('full_name')->all();
|
||||
$this->assertContains('Jan Festivalmedewerker', $personNames);
|
||||
$this->assertNotContains('Piet Dagvrijwilliger', $personNames);
|
||||
}
|
||||
|
||||
@@ -152,13 +152,14 @@ class InvitationTest extends TestCase
|
||||
]);
|
||||
|
||||
$response = $this->postJson("/api/v1/invitations/{$invitation->token}/accept", [
|
||||
'name' => 'New User',
|
||||
'first_name' => 'New',
|
||||
'last_name' => 'User',
|
||||
'password' => 'password123',
|
||||
'password_confirmation' => 'password123',
|
||||
]);
|
||||
|
||||
$response->assertOk();
|
||||
$response->assertJsonStructure(['data' => ['user' => ['id', 'name', 'email'], 'token']]);
|
||||
$response->assertJsonStructure(['data' => ['user' => ['id', 'first_name', 'last_name', 'full_name', 'email'], 'token']]);
|
||||
|
||||
$this->assertDatabaseHas('users', ['email' => 'newuser@test.nl']);
|
||||
$this->assertDatabaseHas('organisation_user', [
|
||||
@@ -204,7 +205,8 @@ class InvitationTest extends TestCase
|
||||
]);
|
||||
|
||||
$response = $this->postJson("/api/v1/invitations/{$invitation->token}/accept", [
|
||||
'name' => 'Test',
|
||||
'first_name' => 'Test',
|
||||
'last_name' => 'User',
|
||||
'password' => 'password123',
|
||||
'password_confirmation' => 'password123',
|
||||
]);
|
||||
@@ -222,7 +224,8 @@ class InvitationTest extends TestCase
|
||||
]);
|
||||
|
||||
$response = $this->postJson("/api/v1/invitations/{$invitation->token}/accept", [
|
||||
'name' => 'Test',
|
||||
'first_name' => 'Test',
|
||||
'last_name' => 'User',
|
||||
'password' => 'password123',
|
||||
'password_confirmation' => 'password123',
|
||||
]);
|
||||
|
||||
@@ -119,17 +119,19 @@ class PersonTest extends TestCase
|
||||
|
||||
$response = $this->postJson("/api/v1/events/{$this->event->id}/persons", [
|
||||
'crowd_type_id' => $this->crowdType->id,
|
||||
'name' => 'Jan de Vries',
|
||||
'first_name' => 'Jan',
|
||||
'last_name' => 'de Vries',
|
||||
'email' => 'jan@test.nl',
|
||||
'phone' => '0612345678',
|
||||
]);
|
||||
|
||||
$response->assertCreated()
|
||||
->assertJson(['data' => ['name' => 'Jan de Vries', 'email' => 'jan@test.nl', 'status' => 'pending']]);
|
||||
->assertJson(['data' => ['first_name' => 'Jan', 'last_name' => 'de Vries', 'email' => 'jan@test.nl', 'status' => 'pending']]);
|
||||
|
||||
$this->assertDatabaseHas('persons', [
|
||||
'event_id' => $this->event->id,
|
||||
'name' => 'Jan de Vries',
|
||||
'first_name' => 'Jan',
|
||||
'last_name' => 'de Vries',
|
||||
'email' => 'jan@test.nl',
|
||||
]);
|
||||
}
|
||||
|
||||
@@ -66,7 +66,8 @@ class PersonIdentityMatchTest extends TestCase
|
||||
|
||||
$response = $this->postJson("/api/v1/events/{$this->event->id}/persons", [
|
||||
'crowd_type_id' => $this->crowdType->id,
|
||||
'name' => 'Jan de Vries',
|
||||
'first_name' => 'Jan',
|
||||
'last_name' => 'de Vries',
|
||||
'email' => 'jan@example.nl',
|
||||
]);
|
||||
|
||||
@@ -93,7 +94,8 @@ class PersonIdentityMatchTest extends TestCase
|
||||
|
||||
$response = $this->postJson("/api/v1/events/{$this->event->id}/persons", [
|
||||
'crowd_type_id' => $this->crowdType->id,
|
||||
'name' => 'Piet Jansen',
|
||||
'first_name' => 'Piet',
|
||||
'last_name' => 'Jansen',
|
||||
'email' => 'unknown@example.nl',
|
||||
]);
|
||||
|
||||
|
||||
Reference in New Issue
Block a user