mirror of
https://github.com/VueFileManager/vuefilemanager.git
synced 2026-04-17 15:52:15 +00:00
updating team members/invitations
This commit is contained in:
@@ -1,9 +1,9 @@
|
||||
<?php
|
||||
|
||||
use Domain\Teams\Controllers\BrowseSharedWithMeController;
|
||||
use Domain\Teams\Controllers\ConvertFolderIntoTeamFolderController;
|
||||
use Domain\Teams\Controllers\InvitationsController;
|
||||
use Domain\Teams\Controllers\TeamFoldersController;
|
||||
use Domain\Teams\Controllers\BrowseSharedWithMeController;
|
||||
use Domain\Teams\Controllers\ConvertFolderIntoTeamFolderController;
|
||||
|
||||
Route::apiResource('/invitations', InvitationsController::class);
|
||||
Route::apiResource('/folders', TeamFoldersController::class);
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
<?php
|
||||
|
||||
namespace Domain\Folders\Models;
|
||||
|
||||
use App\Users\Models\User;
|
||||
use Illuminate\Support\Str;
|
||||
use Domain\Files\Models\File;
|
||||
use Laravel\Scout\Searchable;
|
||||
@@ -10,11 +10,13 @@ use Kyslik\ColumnSortable\Sortable;
|
||||
use Database\Factories\FolderFactory;
|
||||
use Illuminate\Database\Eloquent\Model;
|
||||
use TeamTNT\TNTSearch\Indexer\TNTIndexer;
|
||||
use Domain\Teams\Models\TeamFolderInvitation;
|
||||
use \Illuminate\Database\Eloquent\SoftDeletes;
|
||||
use Illuminate\Database\Eloquent\Relations\HasOne;
|
||||
use Illuminate\Database\Eloquent\Relations\HasMany;
|
||||
use Illuminate\Database\Eloquent\Relations\BelongsTo;
|
||||
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
||||
use Illuminate\Database\Eloquent\Relations\BelongsToMany;
|
||||
|
||||
/**
|
||||
* @method static whereUserId(int|string|null $id)
|
||||
@@ -134,9 +136,9 @@ class Folder extends Model
|
||||
/**
|
||||
* Format deleted at date reformat
|
||||
*/
|
||||
public function getDeletedAtAttribute(): string|null
|
||||
public function getDeletedAtAttribute(): string | null
|
||||
{
|
||||
if (!$this->attributes['deleted_at']) {
|
||||
if (! $this->attributes['deleted_at']) {
|
||||
return null;
|
||||
}
|
||||
|
||||
@@ -222,13 +224,24 @@ class Folder extends Model
|
||||
return $this->hasOne(Share::class, 'item_id', 'id');
|
||||
}
|
||||
|
||||
public function teamInvitations(): HasMany
|
||||
{
|
||||
return $this->hasMany(TeamFolderInvitation::class, 'folder_id', 'id');
|
||||
}
|
||||
|
||||
public function teamMembers(): BelongsToMany
|
||||
{
|
||||
return $this->belongsToMany(User::class, 'team_folder_members', 'folder_id', 'user_id')
|
||||
->withPivot('permission');
|
||||
}
|
||||
|
||||
// Delete all folder children
|
||||
public static function boot()
|
||||
{
|
||||
parent::boot();
|
||||
|
||||
static::creating(function ($model) {
|
||||
$model->id = (string)Str::uuid();
|
||||
$model->id = (string) Str::uuid();
|
||||
});
|
||||
|
||||
static::deleting(function ($item) {
|
||||
|
||||
@@ -1,12 +1,10 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Domain\Teams\Actions;
|
||||
|
||||
use Domain\Folders\Models\Folder;
|
||||
use Illuminate\Support\Facades\Notification;
|
||||
use Domain\Teams\Models\TeamFolderInvitation;
|
||||
use Domain\Teams\Notifications\InvitationIntoTeamFolder;
|
||||
use Illuminate\Support\Facades\Notification;
|
||||
|
||||
class InviteMembersIntoTeamFolderAction
|
||||
{
|
||||
@@ -16,7 +14,6 @@ class InviteMembersIntoTeamFolderAction
|
||||
): void {
|
||||
collect($members)
|
||||
->each(function ($member) use ($folder) {
|
||||
|
||||
// Create invitation
|
||||
$invitation = TeamFolderInvitation::create([
|
||||
'permission' => $member['permission'],
|
||||
@@ -29,4 +26,4 @@ class InviteMembersIntoTeamFolderAction
|
||||
->notify(new InvitationIntoTeamFolder($folder, $invitation));
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
57
src/Domain/Teams/Actions/UpdateInvitationsAction.php
Normal file
57
src/Domain/Teams/Actions/UpdateInvitationsAction.php
Normal file
@@ -0,0 +1,57 @@
|
||||
<?php
|
||||
namespace Domain\Teams\Actions;
|
||||
|
||||
use Domain\Folders\Models\Folder;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
|
||||
class UpdateInvitationsAction
|
||||
{
|
||||
public function __construct(
|
||||
public InviteMembersIntoTeamFolderAction $inviteMembers,
|
||||
) {
|
||||
}
|
||||
|
||||
public function __invoke(Folder $folder, $invitations): void
|
||||
{
|
||||
// Get stored invitations from team folder
|
||||
$storedInvitations = $folder
|
||||
->teamInvitations()
|
||||
->pluck('email');
|
||||
|
||||
// Get newbies added by user in request
|
||||
$newbies = collect($invitations)
|
||||
->filter(
|
||||
fn ($invitation) => ! in_array($invitation['email'], $storedInvitations->toArray())
|
||||
);
|
||||
|
||||
// Get deleted invitations by user in request
|
||||
$removed = $storedInvitations->diff(
|
||||
collect($invitations)->pluck('email')->toArray()
|
||||
);
|
||||
|
||||
// Invite team members
|
||||
if ($newbies->isNotEmpty()) {
|
||||
($this->inviteMembers)($newbies->toArray(), $folder);
|
||||
}
|
||||
|
||||
// Delete invite from team folder
|
||||
if ($removed->isNotEmpty()) {
|
||||
DB::table('team_folder_invitations')
|
||||
->where('folder_id', $folder->id)
|
||||
->whereIn('email', $removed)
|
||||
->delete();
|
||||
}
|
||||
|
||||
// Update privileges
|
||||
collect($invitations)
|
||||
->each(
|
||||
fn ($invitation) =>
|
||||
DB::table('team_folder_invitations')
|
||||
->where('folder_id', $folder->id)
|
||||
->where('email', $invitation['email'])
|
||||
->update([
|
||||
'permission' => $invitation['permission'],
|
||||
])
|
||||
);
|
||||
}
|
||||
}
|
||||
40
src/Domain/Teams/Actions/UpdateMembersAction.php
Normal file
40
src/Domain/Teams/Actions/UpdateMembersAction.php
Normal file
@@ -0,0 +1,40 @@
|
||||
<?php
|
||||
namespace Domain\Teams\Actions;
|
||||
|
||||
use DB;
|
||||
use Domain\Folders\Models\Folder;
|
||||
|
||||
class UpdateMembersAction
|
||||
{
|
||||
public function __invoke(Folder $folder, $members): void
|
||||
{
|
||||
$existingMembers = $folder
|
||||
->teamMembers()
|
||||
->pluck('user_id');
|
||||
|
||||
// Get deleted members from request
|
||||
$deletedMembers = $existingMembers->diff(
|
||||
collect($members)->pluck('id')->toArray()
|
||||
);
|
||||
|
||||
// Remove team members from team folder
|
||||
if ($deletedMembers->isNotEmpty()) {
|
||||
DB::table('team_folder_members')
|
||||
->where('folder_id', $folder->id)
|
||||
->whereIn('user_id', $deletedMembers->toArray())
|
||||
->delete();
|
||||
}
|
||||
|
||||
// Update privileges
|
||||
collect($members)
|
||||
->each(
|
||||
fn ($member) =>
|
||||
DB::table('team_folder_members')
|
||||
->where('folder_id', $folder->id)
|
||||
->where('user_id', $member['id'])
|
||||
->update([
|
||||
'permission' => $member['permission'],
|
||||
])
|
||||
);
|
||||
}
|
||||
}
|
||||
@@ -1,14 +1,11 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Domain\Teams\Controllers;
|
||||
|
||||
|
||||
use Str;
|
||||
use Domain\Files\Models\File;
|
||||
use Domain\Folders\Models\Folder;
|
||||
use Illuminate\Support\Facades\Auth;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
use Str;
|
||||
use Illuminate\Support\Facades\Auth;
|
||||
|
||||
class BrowseSharedWithMeController
|
||||
{
|
||||
@@ -30,7 +27,6 @@ class BrowseSharedWithMeController
|
||||
}
|
||||
|
||||
if (! $rootId) {
|
||||
|
||||
$folderIds = DB::table('team_folder_members')
|
||||
->where('user_id', Auth::id())
|
||||
->pluck('folder_id');
|
||||
@@ -49,4 +45,4 @@ class BrowseSharedWithMeController
|
||||
'folder' => $requestedFolder,
|
||||
];
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,26 +1,23 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Domain\Teams\Controllers;
|
||||
|
||||
|
||||
use App\Http\Controllers\Controller;
|
||||
use Domain\Folders\Models\Folder;
|
||||
use Domain\Teams\Actions\InviteMembersIntoTeamFolderAction;
|
||||
use Illuminate\Http\Request;
|
||||
use Illuminate\Http\Response;
|
||||
use Domain\Folders\Models\Folder;
|
||||
use App\Http\Controllers\Controller;
|
||||
use Domain\Teams\Actions\InviteMembersIntoTeamFolderAction;
|
||||
|
||||
class ConvertFolderIntoTeamFolderController extends Controller
|
||||
{
|
||||
public function __construct(
|
||||
public InviteMembersIntoTeamFolderAction $inviteMembers,
|
||||
) {}
|
||||
) {
|
||||
}
|
||||
|
||||
public function __invoke(
|
||||
Request $request,
|
||||
Folder $folder
|
||||
): Response {
|
||||
|
||||
$folder->update([
|
||||
'team_folder' => 1,
|
||||
'parent_id' => null,
|
||||
@@ -31,4 +28,4 @@ class ConvertFolderIntoTeamFolderController extends Controller
|
||||
|
||||
return response($folder, 201);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,12 +1,11 @@
|
||||
<?php
|
||||
|
||||
namespace Domain\Teams\Controllers;
|
||||
|
||||
use App\Http\Controllers\Controller;
|
||||
use App\Users\Models\User;
|
||||
use Domain\Teams\Models\TeamFolderInvitation;
|
||||
use Illuminate\Http\Response;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
use App\Http\Controllers\Controller;
|
||||
use Domain\Teams\Models\TeamFolderInvitation;
|
||||
|
||||
class InvitationsController extends Controller
|
||||
{
|
||||
@@ -23,7 +22,7 @@ class InvitationsController extends Controller
|
||||
DB::table('team_folder_members')
|
||||
->insert([
|
||||
'folder_id' => $invitation->folder_id,
|
||||
'user_id' => $user->id,
|
||||
'user_id' => $user->id,
|
||||
'permission' => 'can-edit',
|
||||
]);
|
||||
|
||||
@@ -33,11 +32,10 @@ class InvitationsController extends Controller
|
||||
public function destroy(
|
||||
TeamFolderInvitation $invitation
|
||||
): Response {
|
||||
|
||||
$invitation->update([
|
||||
'status' => 'rejected',
|
||||
]);
|
||||
|
||||
return response('Done', 204);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,23 +1,24 @@
|
||||
<?php
|
||||
|
||||
namespace Domain\Teams\Controllers;
|
||||
|
||||
use DB;
|
||||
use Domain\Files\Models\File;
|
||||
use Domain\Teams\Actions\InviteMembersIntoTeamFolderAction;
|
||||
use Illuminate\Http\Request;
|
||||
use Domain\Files\Models\File;
|
||||
use Illuminate\Http\Response;
|
||||
use Domain\Folders\Models\Folder;
|
||||
use App\Http\Controllers\Controller;
|
||||
use Domain\Teams\DTO\CreateTeamFolderData;
|
||||
use Illuminate\Support\Facades\Auth;
|
||||
use Illuminate\Support\Str;
|
||||
use Domain\Teams\DTO\CreateTeamFolderData;
|
||||
use Domain\Teams\Actions\UpdateMembersAction;
|
||||
use Domain\Teams\Actions\UpdateInvitationsAction;
|
||||
use Domain\Teams\Actions\InviteMembersIntoTeamFolderAction;
|
||||
|
||||
class TeamFoldersController extends Controller
|
||||
{
|
||||
public function __construct(
|
||||
public InviteMembersIntoTeamFolderAction $inviteMembers,
|
||||
) {}
|
||||
) {
|
||||
}
|
||||
|
||||
public function show($id)
|
||||
{
|
||||
@@ -62,62 +63,41 @@ class TeamFoldersController extends Controller
|
||||
return response($folder, 201);
|
||||
}
|
||||
|
||||
public function update(Request $request, Folder $folder)
|
||||
{
|
||||
$existingMembers = DB::table('team_folder_members')
|
||||
->where('folder_id', $folder->id)
|
||||
->pluck('user_id');
|
||||
public function update(
|
||||
Request $request,
|
||||
Folder $folder,
|
||||
UpdateInvitationsAction $updateInvitations,
|
||||
UpdateMembersAction $updateMembers,
|
||||
): Response {
|
||||
$updateInvitations(
|
||||
$folder,
|
||||
$request->input('invitations')
|
||||
);
|
||||
|
||||
// Get deleted members from request
|
||||
// TODO: vymazat uzivatela z pozvankou
|
||||
$deletedMembers = $existingMembers
|
||||
->filter(fn ($memberId) => ! in_array(
|
||||
$memberId, collect($request->input('members'))->pluck('id')->toArray()
|
||||
));
|
||||
|
||||
// Get newly added members from request
|
||||
$newMembers = collect($request->input('members'))
|
||||
->filter(fn ($member) => ! Str::isUuid($member['id']));
|
||||
|
||||
// Invite team members
|
||||
if ($newMembers->isNotEmpty()) {
|
||||
($this->inviteMembers)($newMembers->toArray(), $folder);
|
||||
}
|
||||
|
||||
// Remove team members from team folder
|
||||
if ($deletedMembers->isNotEmpty()) {
|
||||
DB::table('team_folder_members')
|
||||
->whereIn('user_id', $deletedMembers->toArray())
|
||||
->delete();
|
||||
}
|
||||
|
||||
// Update privileges
|
||||
collect($request->input('members'))
|
||||
->each(function ($member) {
|
||||
DB::table('team_folder_members')
|
||||
->where('user_id', $member['id'])
|
||||
->update([
|
||||
'permission' => $member['permission'],
|
||||
]);
|
||||
});
|
||||
$updateMembers(
|
||||
$folder,
|
||||
$request->input('members')
|
||||
);
|
||||
|
||||
return response('Done', 201);
|
||||
}
|
||||
|
||||
public function destroy(Folder $folder): Response
|
||||
{
|
||||
$folder->update([
|
||||
'team_folder' => 0,
|
||||
]);
|
||||
|
||||
// Delete existing invitations
|
||||
DB::table('team_folder_invitations')
|
||||
->where('folder_id', $folder->id)
|
||||
->delete();
|
||||
|
||||
// Delete attached members from folder
|
||||
DB::table('team_folder_members')
|
||||
->where('folder_id', $folder->id)
|
||||
->delete();
|
||||
|
||||
$folder->update([
|
||||
'team_folder' => 0,
|
||||
]);
|
||||
|
||||
return response('Done.', 204);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,14 +1,13 @@
|
||||
<?php
|
||||
|
||||
namespace Tests\Domain\Teams;
|
||||
|
||||
use Domain\Files\Models\File;
|
||||
use Domain\Folders\Models\Folder;
|
||||
use Domain\Teams\Models\TeamFolderInvitation;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
use Notification;
|
||||
use Tests\TestCase;
|
||||
use App\Users\Models\User;
|
||||
use Domain\Files\Models\File;
|
||||
use Domain\Folders\Models\Folder;
|
||||
use Illuminate\Support\Facades\DB;
|
||||
use Domain\Teams\Models\TeamFolderInvitation;
|
||||
use Domain\Teams\Notifications\InvitationIntoTeamFolder;
|
||||
|
||||
class TeamsTest extends TestCase
|
||||
@@ -179,7 +178,7 @@ class TeamsTest extends TestCase
|
||||
/**
|
||||
* @test
|
||||
*/
|
||||
public function it_add_member_into_team_folder()
|
||||
public function it_invite_member_into_team_folder()
|
||||
{
|
||||
$user = User::factory(User::class)
|
||||
->create();
|
||||
@@ -194,6 +193,14 @@ class TeamsTest extends TestCase
|
||||
'team_folder' => 1,
|
||||
]);
|
||||
|
||||
TeamFolderInvitation::factory()
|
||||
->create([
|
||||
'folder_id' => $folder->id,
|
||||
'status' => 'pending',
|
||||
'permission' => 'can-edit',
|
||||
'email' => 'existing@member.com',
|
||||
]);
|
||||
|
||||
DB::table('team_folder_members')
|
||||
->insert([
|
||||
[
|
||||
@@ -211,7 +218,96 @@ class TeamsTest extends TestCase
|
||||
$this
|
||||
->actingAs($user)
|
||||
->patchJson("/api/teams/folders/{$folder->id}", [
|
||||
'members' => [
|
||||
'members' => [
|
||||
[
|
||||
'id' => $members[0]->id,
|
||||
'email' => 'john@internal.com',
|
||||
'permission' => 'can-edit',
|
||||
],
|
||||
[
|
||||
'id' => $members[1]->id,
|
||||
'email' => 'jane@external.com',
|
||||
'permission' => 'can-edit',
|
||||
],
|
||||
],
|
||||
'invitations' => [
|
||||
[
|
||||
'id' => null,
|
||||
'email' => 'existing@member.com',
|
||||
'permission' => 'can-edit',
|
||||
],
|
||||
[
|
||||
'id' => null,
|
||||
'email' => 'added@member.com',
|
||||
'permission' => 'can-view',
|
||||
],
|
||||
],
|
||||
])
|
||||
->assertCreated();
|
||||
|
||||
$this
|
||||
->assertDatabaseCount('team_folder_members', 2)
|
||||
->assertDatabaseCount('team_folder_invitations', 2)
|
||||
->assertDatabaseHas('team_folder_invitations', [
|
||||
'email' => 'added@member.com',
|
||||
'permission' => 'can-view',
|
||||
]);
|
||||
|
||||
Notification::assertTimesSent(1, InvitationIntoTeamFolder::class);
|
||||
}
|
||||
|
||||
/**
|
||||
* @test
|
||||
*/
|
||||
public function it_delete_invited_member_from_team_folder()
|
||||
{
|
||||
$user = User::factory(User::class)
|
||||
->create();
|
||||
|
||||
$members = User::factory(User::class)
|
||||
->count(2)
|
||||
->create();
|
||||
|
||||
$folder = Folder::factory()
|
||||
->create([
|
||||
'user_id' => $user->id,
|
||||
'team_folder' => 1,
|
||||
]);
|
||||
|
||||
TeamFolderInvitation::factory()
|
||||
->create([
|
||||
'folder_id' => $folder->id,
|
||||
'status' => 'pending',
|
||||
'permission' => 'can-edit',
|
||||
'email' => 'deleted@member.com',
|
||||
]);
|
||||
|
||||
TeamFolderInvitation::factory()
|
||||
->create([
|
||||
'folder_id' => $folder->id,
|
||||
'status' => 'pending',
|
||||
'permission' => 'can-edit',
|
||||
'email' => 'existing@member.com',
|
||||
]);
|
||||
|
||||
DB::table('team_folder_members')
|
||||
->insert([
|
||||
[
|
||||
'folder_id' => $folder->id,
|
||||
'user_id' => $members[0]->id,
|
||||
'permission' => 'can-edit',
|
||||
],
|
||||
[
|
||||
'folder_id' => $folder->id,
|
||||
'user_id' => $members[1]->id,
|
||||
'permission' => 'can-edit',
|
||||
],
|
||||
]);
|
||||
|
||||
$this
|
||||
->actingAs($user)
|
||||
->patchJson("/api/teams/folders/{$folder->id}", [
|
||||
'members' => [
|
||||
[
|
||||
'id' => $members[0]->id,
|
||||
'email' => 'john@internal.com',
|
||||
@@ -222,9 +318,11 @@ class TeamsTest extends TestCase
|
||||
'email' => 'jane@external.com',
|
||||
'permission' => 'can-view',
|
||||
],
|
||||
],
|
||||
'invitations' => [
|
||||
[
|
||||
'id' => null,
|
||||
'email' => 'new@member.com',
|
||||
'email' => 'existing@member.com',
|
||||
'permission' => 'can-view',
|
||||
],
|
||||
],
|
||||
@@ -233,12 +331,10 @@ class TeamsTest extends TestCase
|
||||
|
||||
$this
|
||||
->assertDatabaseCount('team_folder_members', 2)
|
||||
->assertDatabaseCount('team_folder_invitations', 1)
|
||||
->assertDatabaseHas('team_folder_invitations', [
|
||||
'email' => 'new@member.com',
|
||||
'permission' => 'can-view',
|
||||
'email' => 'existing@member.com',
|
||||
]);
|
||||
|
||||
Notification::assertTimesSent(1, InvitationIntoTeamFolder::class);
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -283,6 +379,7 @@ class TeamsTest extends TestCase
|
||||
'permission' => 'can-edit',
|
||||
],
|
||||
],
|
||||
'invitations' => [],
|
||||
])
|
||||
->assertCreated();
|
||||
|
||||
@@ -293,6 +390,53 @@ class TeamsTest extends TestCase
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* @test
|
||||
*/
|
||||
public function it_update_invited_member_permission_in_team_folder()
|
||||
{
|
||||
$user = User::factory(User::class)
|
||||
->create();
|
||||
|
||||
$folder = Folder::factory()
|
||||
->create([
|
||||
'user_id' => $user->id,
|
||||
'team_folder' => 1,
|
||||
]);
|
||||
|
||||
TeamFolderInvitation::factory()
|
||||
->create([
|
||||
'folder_id' => $folder->id,
|
||||
'status' => 'pending',
|
||||
'permission' => 'can-view',
|
||||
'email' => 'existing@member.com',
|
||||
]);
|
||||
|
||||
$this
|
||||
->actingAs($user)
|
||||
->patchJson("/api/teams/folders/{$folder->id}", [
|
||||
'members' => [],
|
||||
'invitations' => [
|
||||
[
|
||||
'id' => null,
|
||||
'email' => 'existing@member.com',
|
||||
'permission' => 'can-edit',
|
||||
],
|
||||
],
|
||||
])
|
||||
->assertCreated();
|
||||
|
||||
$this
|
||||
->assertDatabaseCount('team_folder_members', 0)
|
||||
->assertDatabaseCount('team_folder_invitations', 1)
|
||||
->assertDatabaseHas('team_folder_invitations', [
|
||||
'email' => 'existing@member.com',
|
||||
'permission' => 'can-edit',
|
||||
]);
|
||||
|
||||
Notification::assertTimesSent(0, InvitationIntoTeamFolder::class);
|
||||
}
|
||||
|
||||
/**
|
||||
* @test
|
||||
*/
|
||||
@@ -414,7 +558,7 @@ class TeamsTest extends TestCase
|
||||
|
||||
$this
|
||||
->actingAs($user)
|
||||
->getJson("/api/teams/folders/undefined")
|
||||
->getJson('/api/teams/folders/undefined')
|
||||
->assertOk()
|
||||
->assertJsonFragment([
|
||||
'id' => $folder->id,
|
||||
|
||||
Reference in New Issue
Block a user