feat: complete person identity matching system with fuzzy detection, revert, and manual link

Implements the full identity matching engine: email matching (HIGH confidence),
fuzzy name matching with Levenshtein distance (MEDIUM confidence, upgradable to
HIGH with DOB tiebreaker), manual link/unlink, revert confirmed matches, and
automatic detection via PersonObserver. Includes 33 comprehensive tests, frontend
integration with confirm/dismiss/unlink UI, and match indicators in the persons list.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-04-14 08:44:24 +02:00
parent 7932e53daf
commit eb1a0ac666
30 changed files with 1941 additions and 399 deletions

View File

@@ -17,6 +17,7 @@ final class MeResource extends JsonResource
'first_name' => $this->first_name,
'last_name' => $this->last_name,
'full_name' => $this->full_name,
'date_of_birth' => $this->date_of_birth?->toDateString(),
'email' => $this->email,
'timezone' => $this->timezone,
'locale' => $this->locale,

View File

@@ -27,18 +27,32 @@ final class PersonIdentityMatchResource extends JsonResource
],
'matched_user' => [
'id' => $this->matchedUser->id,
'name' => $this->matchedUser->name,
'first_name' => $this->matchedUser->first_name,
'last_name' => $this->matchedUser->last_name,
'full_name' => $this->matchedUser->full_name,
'email' => $this->matchedUser->email,
'date_of_birth' => $this->matchedUser->date_of_birth?->toDateString(),
],
'matched_on' => $this->matched_on->value,
'matched_on_label' => $this->matched_on->label(),
'confidence' => $this->confidence->value,
'confidence_label' => $this->confidence->label(),
'status' => $this->status->value,
'status_label' => $this->status->label(),
'match_details' => $this->match_details,
'confirmed_by' => $this->when($this->confirmedBy, fn () => [
'id' => $this->confirmedBy->id,
'full_name' => $this->confirmedBy->full_name,
]),
'confirmed_at' => $this->confirmed_at?->toIso8601String(),
'dismissed_at' => $this->dismissed_at?->toIso8601String(),
'reverted_at' => $this->reverted_at?->toIso8601String(),
'resolved_by' => $this->when($this->resolvedBy, fn () => [
'id' => $this->resolvedBy->id,
'name' => $this->resolvedBy->name,
'full_name' => $this->resolvedBy->full_name,
]),
'resolved_at' => $this->resolved_at?->toISOString(),
'created_at' => $this->created_at->toISOString(),
'resolved_at' => $this->resolved_at?->toIso8601String(),
'created_at' => $this->created_at->toIso8601String(),
];
}
}

View File

@@ -28,6 +28,12 @@ final class PersonResource extends JsonResource
'created_at' => $this->created_at->toIso8601String(),
'crowd_type' => new CrowdTypeResource($this->whenLoaded('crowdType')),
'company' => new CompanyResource($this->whenLoaded('company')),
'has_user_account' => (bool) $this->user_id,
'user_account' => $this->when($this->user_id && $this->relationLoaded('user') && $this->user, fn () => [
'id' => $this->user->id,
'email' => $this->user->email,
'full_name' => $this->user->full_name,
]),
'pending_identity_match' => $this->when(
$this->relationLoaded('pendingIdentityMatch') && $this->pendingIdentityMatch,
function () {
@@ -41,9 +47,13 @@ final class PersonResource extends JsonResource
'last_name' => $match->matchedUser->last_name,
'full_name' => $match->matchedUser->full_name,
'email' => $match->matchedUser->email,
'date_of_birth' => $match->matchedUser->date_of_birth?->toDateString(),
],
'matched_on' => $match->matched_on->value,
'matched_on_label' => $match->matched_on->label(),
'confidence' => $match->confidence->value,
'confidence_label' => $match->confidence->label(),
'match_details' => $match->match_details,
];
}
),

View File

@@ -16,6 +16,7 @@ final class UserResource extends JsonResource
'first_name' => $this->first_name,
'last_name' => $this->last_name,
'full_name' => $this->full_name,
'date_of_birth' => $this->date_of_birth?->toDateString(),
'email' => $this->email,
'roles' => $this->getRoleNames()->values()->all(),
'timezone' => $this->timezone,