rooberthh / identity
A Laravel package to identify and validate Swedish personal numbers (personnummer) and organization numbers (organisationsnummer).
Installs: 0
Dependents: 0
Suggesters: 0
Security: 0
Stars: 0
Watchers: 0
Forks: 0
pkg:composer/rooberthh/identity
Requires
- php: ^8.3
- illuminate/contracts: ^11.0|^12.0
- illuminate/support: ^11.0|^12.0
- illuminate/validation: ^11.0|^12.0
Requires (Dev)
- larastan/larastan: ^3.0
- laravel/pint: ^1.13.7
- pestphp/pest: ^4.3
This package is auto-updated.
Last update: 2026-02-11 17:09:45 UTC
README
A PHP package to validate and identify Swedish personal numbers (personnummer) and organization numbers (organisationsnummer).
Usage
Auto-detect identity type
use Rooberthh\Identity\Identity; $identity = Identity::identify('770604-0016'); // Returns PersonalNumber instance $identity = Identity::identify('556074-7569'); // Returns OrganizationNumber instance
Safe identification (no exceptions)
$identity = Identity::tryIdentify('invalid-number'); // Returns null instead of throwing exception if ($identity = Identity::tryIdentify($input)) { // Valid identity }
Personal number validation
use Rooberthh\Identity\PersonalNumber; // Validate without creating object if (PersonalNumber::isValid('770604-0016')) { // Valid personal number } // Create instance (throws exception if invalid) $person = new PersonalNumber('770604-0016');
Personal number formatting
$person = new PersonalNumber('7706040016'); $person->shortFormat(); // "7706040016" $person->shortFormat(true); // "770604-0016" $person->longFormat(); // "197706040016" $person->longFormat(true); // "19770604-0016"
Extract metadata from personal number
$person = new PersonalNumber('770604-0016'); $person->getBirthDate(); // DateTimeImmutable: 1977-06-04 $person->getAge(); // int (calculated from today) $person->getGender(); // Gender::Male $person->isMale(); // true $person->isFemale(); // false $person->isOfAge(18); // true
Centenarians (100+ years old)
// The + separator indicates a person born 100+ years ago $person = new PersonalNumber('770604+0016'); $person->isCentenarian(); // true $person->getBirthDate(); // DateTimeImmutable: 1877-06-04 $person->shortFormat(true); // "770604+0016"
Coordination numbers (samordningsnummer)
// Coordination numbers have day + 60 $person = new PersonalNumber('770664-0005'); $person->isCoordinationNumber(); // true $person->getBirthDate(); // DateTimeImmutable: 1977-06-04
Organization number validation
use Rooberthh\Identity\OrganizationNumber; if (OrganizationNumber::isValid('556074-7569')) { // Valid organization number } $org = new OrganizationNumber('5560747569'); $org->shortFormat(true); // "556074-7569"
Error handling
use Rooberthh\Identity\Exceptions\IdentityException; try { $person = new PersonalNumber('invalid'); } catch (IdentityException $e) { // "'invalid' is not valid personal number." }
Laravel Usage
Validation Rules
<?php namespace App\Http\Requests; use Illuminate\Foundation\Http\FormRequest; use Rooberthh\Identity\Rules\PersonalNumberRule; use Rooberthh\Identity\Rules\OrganizationNumberRule; use Rooberthh\Identity\Rules\IdentityRule; class StoreCustomerRequest extends FormRequest { public function rules(): array { return [ // Validate personal number only 'personal_number' => ['required', new PersonalNumberRule], // Validate organization number only 'organization_number' => ['required', new OrganizationNumberRule], // Validate either type (personal or organization) 'identity_number' => ['required', new IdentityRule], ]; } }
Model Casts
<?php namespace App\Models; use Illuminate\Database\Eloquent\Model; use Rooberthh\Identity\Casts\PersonalNumberCast; use Rooberthh\Identity\Casts\OrganizationNumberCast; use Rooberthh\Identity\Casts\IdentityCast; class Customer extends Model { protected function casts(): array { return [ 'personal_number' => PersonalNumberCast::class, 'organization_number' => OrganizationNumberCast::class, 'identity_number' => IdentityCast::class, // Auto-detects type ]; } }
Usage with Casted Attributes
$customer = Customer::find(1); // Personal number - returns PersonalNumber object $customer->personal_number->shortFormat(true); // "770604-0016" $customer->personal_number->longFormat(); // "197706040016" $customer->personal_number->getBirthDate(); // DateTimeImmutable $customer->personal_number->getAge(); // 48 $customer->personal_number->getGender(); // Gender::Male $customer->personal_number->isMale(); // true // Organization number - returns OrganizationNumber object $customer->organization_number->shortFormat(true); // "556074-7569" // Identity cast - auto-detects and returns appropriate type if ($customer->identity_number instanceof PersonalNumber) { echo $customer->identity_number->getAge(); } // Setting values - accepts string or object $customer->personal_number = '770604-0016'; $customer->personal_number = new PersonalNumber('770604-0016'); $customer->save(); // Stored as "197706040016" (long format)