mirror of
https://github.com/solidtime-io/solidtime.git
synced 2026-06-15 05:22:44 +01:00
340 lines
12 KiB
PHP
340 lines
12 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Tests\Unit\Endpoint\Api\V1;
|
|
|
|
use App\Enums\Role;
|
|
use App\Http\Controllers\Api\V1\InvitationController;
|
|
use App\Mail\OrganizationInvitationMail;
|
|
use App\Models\Member;
|
|
use App\Models\OrganizationInvitation;
|
|
use App\Models\User;
|
|
use Illuminate\Support\Facades\Mail;
|
|
use Laravel\Passport\Passport;
|
|
use PHPUnit\Framework\Attributes\UsesClass;
|
|
|
|
#[UsesClass(InvitationController::class)]
|
|
class InvitationEndpointTest extends ApiEndpointTestAbstract
|
|
{
|
|
public function test_index_fails_if_user_has_no_permission_to_view_invitations(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission();
|
|
Passport::actingAs($data->user);
|
|
|
|
// Act
|
|
$response = $this->getJson(route('api.v1.invitations.index', $data->organization->id));
|
|
|
|
// Assert
|
|
$response->assertStatus(403);
|
|
}
|
|
|
|
public function test_index_returns_invitations_of_organization(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission([
|
|
'invitations:view',
|
|
]);
|
|
$invitation1 = OrganizationInvitation::factory()->forOrganization($data->organization)->create();
|
|
Passport::actingAs($data->user);
|
|
|
|
// Act
|
|
$response = $this->getJson(route('api.v1.invitations.index', $data->organization->getKey()));
|
|
|
|
// Assert
|
|
$response->assertStatus(200);
|
|
$response->assertJson([
|
|
'data' => [
|
|
[
|
|
'id' => $invitation1->getKey(),
|
|
'email' => $invitation1->email,
|
|
'role' => $invitation1->role,
|
|
],
|
|
],
|
|
]);
|
|
}
|
|
|
|
public function test_index_returns_invitations_ordered_by_created_at_descending(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission([
|
|
'invitations:view',
|
|
]);
|
|
$invitationOldest = OrganizationInvitation::factory()->forOrganization($data->organization)->create([
|
|
'created_at' => now()->subDays(3),
|
|
]);
|
|
$invitationNewest = OrganizationInvitation::factory()->forOrganization($data->organization)->create([
|
|
'created_at' => now()->subDay(),
|
|
]);
|
|
$invitationMiddle = OrganizationInvitation::factory()->forOrganization($data->organization)->create([
|
|
'created_at' => now()->subDays(2),
|
|
]);
|
|
Passport::actingAs($data->user);
|
|
|
|
// Act
|
|
$response = $this->getJson(route('api.v1.invitations.index', $data->organization->getKey()));
|
|
|
|
// Assert
|
|
$response->assertStatus(200);
|
|
$ids = collect($response->json('data'))->pluck('id')->values()->toArray();
|
|
$this->assertSame([$invitationNewest->getKey(), $invitationMiddle->getKey(), $invitationOldest->getKey()], $ids);
|
|
}
|
|
|
|
public function test_store_fails_if_user_has_no_permission_to_create_invitations(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission();
|
|
Passport::actingAs($data->user);
|
|
|
|
// Act
|
|
$response = $this->postJson(route('api.v1.invitations.store', $data->organization->getKey()), [
|
|
'email' => 'test@mail.test',
|
|
'role' => Role::Employee->value,
|
|
]);
|
|
|
|
// Assert
|
|
$response->assertStatus(403);
|
|
}
|
|
|
|
public function test_store_fails_if_user_invites_with_role_owner(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission([
|
|
'invitations:create',
|
|
]);
|
|
Passport::actingAs($data->user);
|
|
|
|
// Act
|
|
$response = $this->postJson(route('api.v1.invitations.store', $data->organization->getKey()), [
|
|
'email' => 'test@asdf.at',
|
|
'role' => Role::Owner->value,
|
|
]);
|
|
|
|
// Assert
|
|
$response->assertStatus(422);
|
|
$response->assertJsonPath('message', 'The selected role is invalid.');
|
|
}
|
|
|
|
public function test_store_fails_if_user_invites_with_role_placeholder(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission([
|
|
'invitations:create',
|
|
]);
|
|
Passport::actingAs($data->user);
|
|
|
|
// Act
|
|
$response = $this->postJson(route('api.v1.invitations.store', $data->organization->getKey()), [
|
|
'email' => 'test@asdf.at',
|
|
'role' => Role::Placeholder->value,
|
|
]);
|
|
|
|
// Assert
|
|
$response->assertStatus(422);
|
|
$response->assertJsonPath('message', 'The selected role is invalid.');
|
|
}
|
|
|
|
public function test_store_fails_if_user_invites_user_who_is_already_member_of_organization(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission([
|
|
'invitations:create',
|
|
]);
|
|
Passport::actingAs($data->user);
|
|
$member = Member::factory()->forOrganization($data->organization)->create();
|
|
|
|
// Act
|
|
$response = $this->postJson(route('api.v1.invitations.store', $data->organization->getKey()), [
|
|
'email' => $member->user->email,
|
|
'role' => Role::Employee->value,
|
|
]);
|
|
|
|
// Assert
|
|
$response->assertStatus(400);
|
|
$response->assertJsonPath('message', 'User is already a member of the organization');
|
|
}
|
|
|
|
public function test_store_fails_if_an_invitation_with_the_same_email_already_exists(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission([
|
|
'invitations:create',
|
|
]);
|
|
Passport::actingAs($data->user);
|
|
$email = 'user@email.test';
|
|
$invitation = OrganizationInvitation::factory()->forOrganization($data->organization)->create([
|
|
'email' => $email,
|
|
]);
|
|
|
|
// Act
|
|
$response = $this->postJson(route('api.v1.invitations.store', $data->organization->getKey()), [
|
|
'email' => $email,
|
|
'role' => Role::Employee->value,
|
|
]);
|
|
|
|
// Assert
|
|
$response->assertStatus(400);
|
|
$response->assertExactJson([
|
|
'error' => true,
|
|
'key' => 'invitation_for_the_email_already_exists',
|
|
'message' => 'The email has already been invited to the organization. Please wait for the user to accept the invitation or resend the invitation email.',
|
|
]);
|
|
}
|
|
|
|
public function test_store_works_if_user_invites_user_who_is_also_a_placeholder(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission([
|
|
'invitations:create',
|
|
]);
|
|
$user = User::factory()->placeholder()->create();
|
|
$member = Member::factory()->forOrganization($data->organization)->forUser($user)->role(Role::Placeholder)->create();
|
|
Passport::actingAs($data->user);
|
|
|
|
// Act
|
|
$response = $this->withoutExceptionHandling()->postJson(route('api.v1.invitations.store', $data->organization->getKey()), [
|
|
'email' => $user->email,
|
|
'role' => Role::Employee->value,
|
|
]);
|
|
|
|
// Assert
|
|
$response->assertStatus(204);
|
|
$invitation = OrganizationInvitation::first();
|
|
$this->assertNotNull($invitation);
|
|
$this->assertEquals($user->email, $invitation->email);
|
|
$this->assertEquals(Role::Employee->value, $invitation->role);
|
|
Mail::assertQueued(fn (OrganizationInvitationMail $mail): bool => $mail->invitation->is($invitation));
|
|
Mail::assertNothingSent();
|
|
}
|
|
|
|
public function test_store_invites_user_to_organization(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission([
|
|
'invitations:create',
|
|
]);
|
|
Passport::actingAs($data->user);
|
|
|
|
// Act
|
|
$response = $this->postJson(route('api.v1.invitations.store', $data->organization->getKey()), [
|
|
'email' => 'test@asdf.at',
|
|
'role' => Role::Employee->value,
|
|
]);
|
|
|
|
// Assert
|
|
$response->assertStatus(204);
|
|
$invitation = OrganizationInvitation::first();
|
|
$this->assertNotNull($invitation);
|
|
$this->assertEquals('test@asdf.at', $invitation->email);
|
|
$this->assertEquals(Role::Employee->value, $invitation->role);
|
|
Mail::assertQueued(fn (OrganizationInvitationMail $mail): bool => $mail->invitation->is($invitation));
|
|
Mail::assertNothingSent();
|
|
}
|
|
|
|
public function test_resend_fails_if_user_has_no_permission_to_resend_the_invitation(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission();
|
|
Passport::actingAs($data->user);
|
|
$invitation = OrganizationInvitation::factory()->forOrganization($data->organization)->create();
|
|
|
|
// Act
|
|
$response = $this->postJson(route('api.v1.invitations.resend', [
|
|
$data->organization->getKey(),
|
|
$invitation->getKey(),
|
|
]));
|
|
|
|
// Assert
|
|
Mail::assertNothingSent();
|
|
Mail::assertNothingQueued();
|
|
$response->assertStatus(403);
|
|
}
|
|
|
|
public function test_resend_fails_if_invitation_belongs_to_different_organization(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission([
|
|
'invitations:resend',
|
|
]);
|
|
Passport::actingAs($data->user);
|
|
$invitation = OrganizationInvitation::factory()->create();
|
|
|
|
// Act
|
|
$response = $this->postJson(route('api.v1.invitations.resend', [$data->organization->getKey(), $invitation->getKey()]));
|
|
|
|
// Assert
|
|
Mail::assertNothingSent();
|
|
Mail::assertNothingQueued();
|
|
$response->assertStatus(403);
|
|
}
|
|
|
|
public function test_resend_resends_invitation_email(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission([
|
|
'invitations:resend',
|
|
]);
|
|
Passport::actingAs($data->user);
|
|
$invitation = OrganizationInvitation::factory()->forOrganization($data->organization)->create();
|
|
|
|
// Act
|
|
$response = $this->postJson(route('api.v1.invitations.resend', [
|
|
$data->organization->getKey(),
|
|
$invitation->getKey(),
|
|
]));
|
|
|
|
// Assert
|
|
$response->assertStatus(204);
|
|
Mail::assertQueued(fn (OrganizationInvitationMail $mail): bool => $mail->invitation->is($invitation));
|
|
Mail::assertNothingSent();
|
|
}
|
|
|
|
public function test_delete_fails_if_user_has_no_permission_to_remove_invitations(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission();
|
|
Passport::actingAs($data->user);
|
|
$invitation = OrganizationInvitation::factory()->forOrganization($data->organization)->create();
|
|
|
|
// Act
|
|
$response = $this->deleteJson(route('api.v1.invitations.destroy', [$data->organization->getKey(), $invitation->getKey()]));
|
|
|
|
// Assert
|
|
$response->assertStatus(403);
|
|
}
|
|
|
|
public function test_delete_fails_if_invitation_belongs_to_different_organization(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission([
|
|
'invitations:remove',
|
|
]);
|
|
Passport::actingAs($data->user);
|
|
$invitation = OrganizationInvitation::factory()->create();
|
|
|
|
// Act
|
|
$response = $this->deleteJson(route('api.v1.invitations.destroy', [$data->organization->getKey(), $invitation->getKey()]));
|
|
|
|
// Assert
|
|
$response->assertStatus(403);
|
|
}
|
|
|
|
public function test_delete_removes_invitation(): void
|
|
{
|
|
// Arrange
|
|
$data = $this->createUserWithPermission([
|
|
'invitations:remove',
|
|
]);
|
|
Passport::actingAs($data->user);
|
|
$invitation = OrganizationInvitation::factory()->forOrganization($data->organization)->create();
|
|
|
|
// Act
|
|
$response = $this->deleteJson(route('api.v1.invitations.destroy', [$data->organization->getKey(), $invitation->getKey()]));
|
|
|
|
// Assert
|
|
$response->assertStatus(204);
|
|
$this->assertNull(OrganizationInvitation::find($invitation->getKey()));
|
|
}
|
|
}
|