UserProfileTest: belongs-to user, fillable/non-fillable boundaries, settings cast, lastSubmittedAt accessor (null + max from user-subject submissions only, ignoring drafts and is_test rows). FormSchemaTest: ULID PK, OrganisationScope filtering, polymorphic owner resolution to Event, purpose enum cast, hasMany fields/submissions, and logSchemaChange() actually creates an activity-log entry. FormFieldTest: belongs-to schema, field_type stored as string (not DB enum), binding/translations array casts, hasMany values, soft-delete preserves historical values, logFieldChange() creates an entry. FormSubmissionTest: belongs-to schema, polymorphic subject resolution, status enum cast, schema_snapshot array cast, hasMany values. FormValueTest: belongs-to submission/field, value array cast, hasMany options pivot rebuilt by observer, unique-pair DB constraint enforced. MultiTenancyTest: OrganisationScope correctly filters FormSchema / FormTemplate / FormFieldLibrary by route-resolved organisation. Pins the FormSchemaWebhook un-scoped behaviour explicitly so a future scope addition is an intentional decision, not an accident. MigrationRollbackTest (group 'slow'): full migrate:fresh → rollback 14 S1 steps → assert all 13 form-builder tables dropped + legacy tables intentionally retained → re-migrate and assert table list matches snapshot. Plus a separate test exercising the populate-user-profiles migration's down(). Supporting tweaks: - UserProfile::lastSubmittedAt accessor now returns Carbon|null instead of a raw timestamp string — testable, and matches Eloquent convention. - UserProfileFactory cooperates with UserObserver via newModel override (updates the auto-created row instead of inserting a duplicate). - AppServiceProvider morph map extended with all 12 form-builder model keys so logSchemaChange/logFieldChange resolve under enforceMorphMap. Suite: 945 passed (was 911), 2671 assertions. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
155 lines
6.3 KiB
PHP
155 lines
6.3 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace App\Providers;
|
|
|
|
use App\Models\Company;
|
|
use App\Models\CrowdList;
|
|
use App\Models\CrowdType;
|
|
use App\Models\EmailChangeRequest;
|
|
use App\Models\EmailLog;
|
|
use App\Models\Event;
|
|
use App\Models\FestivalSection;
|
|
use App\Models\ImpersonationSession;
|
|
use App\Models\Location;
|
|
use App\Models\MfaBackupCode;
|
|
use App\Models\MfaEmailCode;
|
|
use App\Models\Organisation;
|
|
use App\Models\OrganisationEmailSettings;
|
|
use App\Models\OrganisationEmailTemplate;
|
|
use App\Models\Person;
|
|
use App\Models\PersonFieldValue;
|
|
use App\Models\PersonIdentityMatch;
|
|
use App\Models\PersonSectionPreference;
|
|
use App\Models\PersonTag;
|
|
use App\Models\RegistrationFieldTemplate;
|
|
use App\Models\RegistrationFormField;
|
|
use App\Models\Shift;
|
|
use App\Models\ShiftAssignment;
|
|
use App\Models\ShiftWaitlist;
|
|
use App\Models\TimeSlot;
|
|
use App\Models\TrustedDevice;
|
|
use App\Models\User;
|
|
use App\Models\UserInvitation;
|
|
use App\Models\UserOrganisationTag;
|
|
use App\Models\UserProfile;
|
|
use App\Models\FormBuilder\FormField;
|
|
use App\Models\FormBuilder\FormFieldLibrary;
|
|
use App\Models\FormBuilder\FormSchema;
|
|
use App\Models\FormBuilder\FormSchemaSection;
|
|
use App\Models\FormBuilder\FormSchemaWebhook;
|
|
use App\Models\FormBuilder\FormSubmission;
|
|
use App\Models\FormBuilder\FormSubmissionDelegation;
|
|
use App\Models\FormBuilder\FormSubmissionSectionStatus;
|
|
use App\Models\FormBuilder\FormTemplate;
|
|
use App\Models\FormBuilder\FormValue;
|
|
use App\Models\FormBuilder\FormValueOption;
|
|
use App\Models\FormBuilder\FormWebhookDelivery;
|
|
use App\Models\VolunteerAvailability;
|
|
use App\Observers\FormBuilder\FormValueObserver;
|
|
use App\Observers\PersonObserver;
|
|
use App\Observers\UserObserver;
|
|
use Illuminate\Auth\Notifications\ResetPassword;
|
|
use Illuminate\Database\Eloquent\Relations\Relation;
|
|
use Illuminate\Support\ServiceProvider;
|
|
use Spatie\Activitylog\Models\Activity;
|
|
|
|
class AppServiceProvider extends ServiceProvider
|
|
{
|
|
public function register(): void
|
|
{
|
|
//
|
|
}
|
|
|
|
public function boot(): void
|
|
{
|
|
// Morph map: explicit keys for every class that can end up in a
|
|
// polymorphic column. Before S1 there were no morphTo/morphMany
|
|
// relations, but spatie/activitylog stores subject/causer as morph
|
|
// columns — so every model passed to performedOn()/causedBy() MUST
|
|
// be registered. Keep form-builder subject_types in sync with
|
|
// config/form_subjects.php.
|
|
Relation::enforceMorphMap([
|
|
// Form-builder subject types
|
|
'event' => Event::class,
|
|
'user' => User::class,
|
|
'user_profile' => UserProfile::class,
|
|
'person' => Person::class,
|
|
'company' => Company::class,
|
|
'organisation' => Organisation::class,
|
|
// 'artist' added when artist module lands
|
|
|
|
// Additional models used as activity-log subjects/causers
|
|
'crowd_list' => CrowdList::class,
|
|
'crowd_type' => CrowdType::class,
|
|
'email_change_request' => EmailChangeRequest::class,
|
|
'email_log' => EmailLog::class,
|
|
'festival_section' => FestivalSection::class,
|
|
'impersonation_session' => ImpersonationSession::class,
|
|
'location' => Location::class,
|
|
'mfa_backup_code' => MfaBackupCode::class,
|
|
'mfa_email_code' => MfaEmailCode::class,
|
|
'organisation_email_settings' => OrganisationEmailSettings::class,
|
|
'organisation_email_template' => OrganisationEmailTemplate::class,
|
|
'person_field_value' => PersonFieldValue::class,
|
|
'person_identity_match' => PersonIdentityMatch::class,
|
|
'person_section_preference' => PersonSectionPreference::class,
|
|
'person_tag' => PersonTag::class,
|
|
'registration_field_template' => RegistrationFieldTemplate::class,
|
|
'registration_form_field' => RegistrationFormField::class,
|
|
'shift' => Shift::class,
|
|
'shift_assignment' => ShiftAssignment::class,
|
|
'shift_waitlist' => ShiftWaitlist::class,
|
|
'time_slot' => TimeSlot::class,
|
|
'trusted_device' => TrustedDevice::class,
|
|
'user_invitation' => UserInvitation::class,
|
|
'user_organisation_tag' => UserOrganisationTag::class,
|
|
'volunteer_availability' => VolunteerAvailability::class,
|
|
|
|
// Form-builder models — used as activity-log subjects via the
|
|
// logSchemaChange / logFieldChange helpers, and (in S2+) as
|
|
// polymorphic webhook payload subjects.
|
|
'form_schema' => FormSchema::class,
|
|
'form_schema_section' => FormSchemaSection::class,
|
|
'form_field' => FormField::class,
|
|
'form_field_library' => FormFieldLibrary::class,
|
|
'form_submission' => FormSubmission::class,
|
|
'form_submission_section_status' => FormSubmissionSectionStatus::class,
|
|
'form_submission_delegation' => FormSubmissionDelegation::class,
|
|
'form_value' => FormValue::class,
|
|
'form_value_option' => FormValueOption::class,
|
|
'form_template' => FormTemplate::class,
|
|
'form_schema_webhook' => FormSchemaWebhook::class,
|
|
'form_webhook_delivery' => FormWebhookDelivery::class,
|
|
]);
|
|
|
|
Person::observe(PersonObserver::class);
|
|
User::observe(UserObserver::class);
|
|
FormValue::observe(FormValueObserver::class);
|
|
|
|
ResetPassword::createUrlUsing(function ($user, string $token) {
|
|
return config('crewli.portal_url') . '/wachtwoord-resetten?token=' . $token . '&email=' . urlencode($user->email);
|
|
});
|
|
|
|
// Tag activity log entries with impersonation context
|
|
Activity::saving(function (Activity $activity) {
|
|
$request = request();
|
|
|
|
$impersonator = $request->attributes->get('impersonator');
|
|
$session = $request->attributes->get('impersonation_session');
|
|
|
|
if ($impersonator && $session) {
|
|
$properties = $activity->properties?->toArray() ?? [];
|
|
$properties['impersonated_by'] = [
|
|
'user_id' => $impersonator->id,
|
|
'name' => $impersonator->full_name,
|
|
'email' => $impersonator->email,
|
|
];
|
|
$properties['impersonation_session_id'] = $session->id;
|
|
$activity->properties = collect($properties);
|
|
}
|
|
});
|
|
}
|
|
}
|