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>
248 lines
7 KiB
PHP
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;
|
|
}
|
|
}
|