Files
relaticle-comments/tests/Feature/MentionParserTest.php
manukminasyan 29fcbd8aec feat: initial release of relaticle/comments
Filament comments package with:
- Polymorphic commenting on any Eloquent model
- Threaded replies with configurable depth
- @mentions with autocomplete and user search
- Emoji reactions with toggle and who-reacted tooltips
- File attachments via Livewire uploads
- Reply and mention notifications via Filament notification system
- Thread subscriptions for notification control
- Real-time broadcasting (opt-in Echo) with polling fallback
- Dark mode support
- CommentsAction, CommentsTableAction, CommentsEntry for Filament integration
- 204 tests, 421 assertions
2026-03-26 23:02:56 +04:00

224 lines
7.1 KiB
PHP

<?php
use Illuminate\Support\Collection;
use Illuminate\Support\Facades\Event;
use Relaticle\Comments\Comment;
use Relaticle\Comments\Contracts\MentionResolver;
use Relaticle\Comments\Events\UserMentioned;
use Relaticle\Comments\Mentions\DefaultMentionResolver;
use Relaticle\Comments\Mentions\MentionParser;
use Relaticle\Comments\Tests\Models\Post;
use Relaticle\Comments\Tests\Models\User;
it('parses @username from plain text body', function () {
User::factory()->create(['name' => 'john']);
User::factory()->create(['name' => 'jane']);
$parser = app(MentionParser::class);
$result = $parser->parse('Hello @john and @jane');
expect($result)->toHaveCount(2);
});
it('parses @username from HTML body', function () {
$john = User::factory()->create(['name' => 'john']);
$parser = app(MentionParser::class);
$result = $parser->parse('<p>Hello @john</p>');
expect($result)->toHaveCount(1);
expect($result->first())->toBe($john->id);
});
it('returns empty collection when no mentions', function () {
$parser = app(MentionParser::class);
$result = $parser->parse('Hello world');
expect($result)->toBeEmpty();
});
it('returns empty collection for non-existent users', function () {
$parser = app(MentionParser::class);
$result = $parser->parse('Hello @ghostuser');
expect($result)->toBeEmpty();
});
it('handles duplicate @mentions', function () {
User::factory()->create(['name' => 'john']);
$parser = app(MentionParser::class);
$result = $parser->parse('@john said hi @john');
expect($result)->toHaveCount(1);
});
it('stores mentions in comment_mentions table on create', function () {
$user = User::factory()->create();
$john = User::factory()->create(['name' => 'john']);
$jane = User::factory()->create(['name' => 'jane']);
$post = Post::factory()->create();
$comment = Comment::factory()->create([
'commentable_id' => $post->id,
'commentable_type' => $post->getMorphClass(),
'user_id' => $user->getKey(),
'user_type' => $user->getMorphClass(),
'body' => '<p>Hello @john and @jane</p>',
]);
app(MentionParser::class)->syncMentions($comment);
expect($comment->mentions()->count())->toBe(2);
expect($comment->mentions->pluck('id')->sort()->values()->all())
->toBe(collect([$john->id, $jane->id])->sort()->values()->all());
});
it('dispatches UserMentioned event for each mentioned user', function () {
Event::fake([UserMentioned::class]);
$user = User::factory()->create();
$john = User::factory()->create(['name' => 'john']);
$jane = User::factory()->create(['name' => 'jane']);
$post = Post::factory()->create();
$comment = Comment::factory()->create([
'commentable_id' => $post->id,
'commentable_type' => $post->getMorphClass(),
'user_id' => $user->getKey(),
'user_type' => $user->getMorphClass(),
'body' => '<p>Hello @john and @jane</p>',
]);
app(MentionParser::class)->syncMentions($comment);
Event::assertDispatched(UserMentioned::class, 2);
Event::assertDispatched(UserMentioned::class, function (UserMentioned $event) use ($comment, $john) {
return $event->comment->id === $comment->id
&& $event->mentionedUser->id === $john->id;
});
Event::assertDispatched(UserMentioned::class, function (UserMentioned $event) use ($comment, $jane) {
return $event->comment->id === $comment->id
&& $event->mentionedUser->id === $jane->id;
});
});
it('only dispatches UserMentioned for newly added mentions on update', function () {
Event::fake([UserMentioned::class]);
$user = User::factory()->create();
$john = User::factory()->create(['name' => 'john']);
$jane = User::factory()->create(['name' => 'jane']);
$post = Post::factory()->create();
$comment = Comment::factory()->create([
'commentable_id' => $post->id,
'commentable_type' => $post->getMorphClass(),
'user_id' => $user->getKey(),
'user_type' => $user->getMorphClass(),
'body' => '<p>Hello @john</p>',
]);
app(MentionParser::class)->syncMentions($comment);
Event::assertDispatched(UserMentioned::class, 1);
Event::fake([UserMentioned::class]);
$comment->update(['body' => '<p>Hello @john and @jane</p>']);
app(MentionParser::class)->syncMentions($comment->fresh());
Event::assertDispatched(UserMentioned::class, 1);
Event::assertDispatched(UserMentioned::class, function (UserMentioned $event) use ($jane) {
return $event->mentionedUser->id === $jane->id;
});
});
it('removes mentions from pivot when user removed from body', function () {
$user = User::factory()->create();
$john = User::factory()->create(['name' => 'john']);
$jane = User::factory()->create(['name' => 'jane']);
$post = Post::factory()->create();
$comment = Comment::factory()->create([
'commentable_id' => $post->id,
'commentable_type' => $post->getMorphClass(),
'user_id' => $user->getKey(),
'user_type' => $user->getMorphClass(),
'body' => '<p>Hello @john and @jane</p>',
]);
app(MentionParser::class)->syncMentions($comment);
expect($comment->mentions()->count())->toBe(2);
$comment->update(['body' => '<p>Hello @john</p>']);
app(MentionParser::class)->syncMentions($comment->fresh());
$comment->refresh();
expect($comment->mentions()->count())->toBe(1);
expect($comment->mentions->first()->id)->toBe($john->id);
});
it('uses configured MentionResolver', function () {
$customResolver = new class implements MentionResolver
{
public function search(string $query): Collection
{
return collect();
}
public function resolveByNames(array $names): Collection
{
return collect();
}
};
$this->app->instance(MentionResolver::class, $customResolver);
$parser = app(MentionParser::class);
$result = $parser->parse('@someuser');
expect($result)->toBeEmpty();
});
it('searches users by name prefix', function () {
User::factory()->create(['name' => 'John Doe']);
User::factory()->create(['name' => 'Jane Doe']);
User::factory()->create(['name' => 'Bob Smith']);
$resolver = app(DefaultMentionResolver::class);
$results = $resolver->search('Jo');
expect($results)->toHaveCount(1);
expect($results->first()->name)->toBe('John Doe');
});
it('limits search results to configured max', function () {
for ($i = 1; $i <= 10; $i++) {
User::factory()->create(['name' => "Test User {$i}"]);
}
config(['comments.mentions.max_results' => 3]);
$resolver = app(DefaultMentionResolver::class);
$results = $resolver->search('Test');
expect($results)->toHaveCount(3);
});
it('resolves users by exact name', function () {
$john = User::factory()->create(['name' => 'John Doe']);
User::factory()->create(['name' => 'Jane Doe']);
$resolver = app(DefaultMentionResolver::class);
$results = $resolver->resolveByNames(['John Doe']);
expect($results)->toHaveCount(1);
expect($results->first()->id)->toBe($john->id);
});