php-tenant/Models/WorkspaceInvitation.php
Claude 126d454bba
chore: add IDE helper annotations to Eloquent models
Add @property, @property-read, @method, and @mixin PHPDoc annotations
to the seven core Eloquent models for IDE autocompletion support.

Models annotated:
- Workspace: all columns, relationships, scopes (active, ordered)
- User: all columns, relationships, factory
- WorkspaceMember: relationship props, scope methods (forWorkspace, forUser, withRole, inTeam, owners)
- WorkspaceInvitation: all columns, relationships, scopes (pending, expired, accepted)
- Namespace_: all columns, relationships, scopes (active, ordered, ownedByUser, ownedByWorkspace, accessibleBy)
- Package: all columns, relationships, scopes (active, public, base, addons, purchasable, free, ordered)
- Feature: all columns, relationships, scopes (active, inCategory, root)

Fixes #31

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-24 13:55:50 +00:00

248 lines
7 KiB
PHP

<?php
declare(strict_types=1);
namespace Core\Tenant\Models;
use Core\Tenant\Database\Factories\WorkspaceInvitationFactory;
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\Relations\BelongsTo;
use Illuminate\Notifications\Notifiable;
use Illuminate\Support\Facades\Hash;
use Illuminate\Support\Str;
/**
* Workspace Invitation - manages invitations to join workspaces.
*
* @property int $id
* @property int $workspace_id
* @property string $email
* @property string $token
* @property string $role
* @property int|null $invited_by
* @property \Illuminate\Support\Carbon $expires_at
* @property \Illuminate\Support\Carbon|null $accepted_at
* @property \Illuminate\Support\Carbon|null $created_at
* @property \Illuminate\Support\Carbon|null $updated_at
* @property-read Workspace $workspace
* @property-read User|null $inviter
* @property-read \Illuminate\Notifications\DatabaseNotificationCollection<int, \Illuminate\Notifications\DatabaseNotification> $notifications
*
* @method static \Illuminate\Database\Eloquent\Builder<static>|WorkspaceInvitation pending()
* @method static \Illuminate\Database\Eloquent\Builder<static>|WorkspaceInvitation expired()
* @method static \Illuminate\Database\Eloquent\Builder<static>|WorkspaceInvitation accepted()
* @method static \Illuminate\Database\Eloquent\Builder<static>|WorkspaceInvitation newModelQuery()
* @method static \Illuminate\Database\Eloquent\Builder<static>|WorkspaceInvitation newQuery()
* @method static \Illuminate\Database\Eloquent\Builder<static>|WorkspaceInvitation query()
* @method static \Core\Tenant\Database\Factories\WorkspaceInvitationFactory factory($count = null, $state = [])
*
* @mixin \Eloquent
*/
class WorkspaceInvitation extends Model
{
use HasFactory;
use Notifiable;
protected static function newFactory(): WorkspaceInvitationFactory
{
return WorkspaceInvitationFactory::new();
}
protected $fillable = [
'workspace_id',
'email',
'token',
'role',
'invited_by',
'expires_at',
'accepted_at',
];
protected $casts = [
'expires_at' => 'datetime',
'accepted_at' => 'datetime',
];
/**
* The "booted" method of the model.
*
* Automatically hashes tokens when creating invitations.
*/
protected static function booted(): void
{
static::creating(function (WorkspaceInvitation $invitation) {
// Only hash if the token looks like a plaintext token (not already hashed)
// Bcrypt hashes start with $2y$ and are 60 chars
if ($invitation->token && ! str_starts_with($invitation->token, '$2y$')) {
$invitation->token = Hash::make($invitation->token);
}
});
}
/**
* Get the workspace this invitation is for.
*/
public function workspace(): BelongsTo
{
return $this->belongsTo(Workspace::class);
}
/**
* Get the user who sent this invitation.
*/
public function inviter(): BelongsTo
{
return $this->belongsTo(User::class, 'invited_by');
}
/**
* Scope to pending invitations (not accepted, not expired).
*/
public function scopePending($query)
{
return $query->whereNull('accepted_at')
->where('expires_at', '>', now());
}
/**
* Scope to expired invitations.
*/
public function scopeExpired($query)
{
return $query->whereNull('accepted_at')
->where('expires_at', '<=', now());
}
/**
* Scope to accepted invitations.
*/
public function scopeAccepted($query)
{
return $query->whereNotNull('accepted_at');
}
/**
* Check if invitation is pending (not accepted and not expired).
*/
public function isPending(): bool
{
return $this->accepted_at === null && $this->expires_at->isFuture();
}
/**
* Check if invitation has expired.
*/
public function isExpired(): bool
{
return $this->accepted_at === null && $this->expires_at->isPast();
}
/**
* Check if invitation has been accepted.
*/
public function isAccepted(): bool
{
return $this->accepted_at !== null;
}
/**
* Generate a unique token for this invitation.
*
* Returns the plaintext token. The token will be hashed when stored.
*/
public static function generateToken(): string
{
// Generate a cryptographically secure random token
// No need to check for uniqueness since hashed tokens are unique
return Str::random(64);
}
/**
* Find invitation by token.
*
* Since tokens are hashed, we must check each pending/valid invitation
* against the provided plaintext token using Hash::check().
*/
public static function findByToken(string $token): ?self
{
// Get all invitations and check the hash
// We limit to recent invitations to improve performance
$invitations = static::orderByDesc('created_at')
->limit(1000)
->get();
foreach ($invitations as $invitation) {
if (Hash::check($token, $invitation->token)) {
return $invitation;
}
}
return null;
}
/**
* Find pending invitation by token.
*
* Since tokens are hashed, we must check each pending invitation
* against the provided plaintext token using Hash::check().
*/
public static function findPendingByToken(string $token): ?self
{
// Get pending invitations and check the hash
$invitations = static::pending()->get();
foreach ($invitations as $invitation) {
if (Hash::check($token, $invitation->token)) {
return $invitation;
}
}
return null;
}
/**
* Verify if the given plaintext token matches this invitation's hashed token.
*/
public function verifyToken(string $plaintextToken): bool
{
return Hash::check($plaintextToken, $this->token);
}
/**
* Accept the invitation for a user.
*/
public function accept(User $user): bool
{
if (! $this->isPending()) {
return false;
}
// Check if user already belongs to this workspace
if ($this->workspace->users()->where('user_id', $user->id)->exists()) {
// Mark as accepted but don't add again
$this->update(['accepted_at' => now()]);
return true;
}
// Add user to workspace with the invited role
$this->workspace->users()->attach($user->id, [
'role' => $this->role,
'is_default' => false,
]);
// Mark invitation as accepted
$this->update(['accepted_at' => now()]);
return true;
}
/**
* Get the notification routing for mail.
*/
public function routeNotificationForMail(): string
{
return $this->email;
}
}