delete items in file request

This commit is contained in:
Čarodej
2022-02-23 10:05:09 +01:00
parent fdfb5f57d7
commit aef715e061
11 changed files with 331 additions and 30 deletions

View File

@@ -1,6 +1,6 @@
APP_NAME=Laravel APP_NAME=Laravel
APP_ENV=local APP_ENV=local
APP_KEY=base64:uM4NZoG/Wgk0ruC0hs4FkYVEChkviOQy8ekOaVpr/pA= APP_KEY=base64:GhP7Rdqy8uICwxrAgwO0pf6xvWIaB/DDo6z2Ro0vIFg=
APP_DEBUG=true APP_DEBUG=true
APP_URL=http://localhost APP_URL=http://localhost
APP_DEMO=false APP_DEMO=false

View File

@@ -35,9 +35,7 @@ class FileFactory extends Factory
'author' => $this->faker->randomElement( 'author' => $this->faker->randomElement(
['user', 'member', 'visitor'] ['user', 'member', 'visitor']
), ),
'created_at' => $this->faker->dateTimeBetween( 'created_at' => $this->faker->dateTimeBetween('-36 months'),
$startDate = '-36 months', $endDate = 'now', $timezone = null
),
]; ];
} }
} }

View File

@@ -313,14 +313,6 @@ const actions = {
if (getters.permission === 'master') { if (getters.permission === 'master') {
if (data.data.type === 'folder') commit('REMOVE_ITEM_FROM_FAVOURITES', data) if (data.data.type === 'folder') commit('REMOVE_ITEM_FROM_FAVOURITES', data)
} }
// Remove file
commit('REMOVE_ITEM', data.data.id)
// Remove item from sidebar
if (getters.permission === 'master') {
if (data.data.type === 'folder') commit('REMOVE_ITEM_FROM_FAVOURITES', data)
}
}) })
// Remove file preview // Remove file preview
@@ -329,7 +321,10 @@ const actions = {
} }
// Get route // Get route
let route = getters.sharedDetail ? `/api/editor/remove/${router.currentRoute.params.token}` : '/api/remove' let route = {
RequestUpload: `/api/upload-request/${router.currentRoute.params.token}/remove`,
Public: `/api/editor/remove/${router.currentRoute.params.token}`,
}[router.currentRoute.name] || '/api/remove'
axios axios
.post(route, { .post(route, {

View File

@@ -1,6 +1,7 @@
<?php <?php
use Domain\UploadRequest\Controllers\CreateFolderController; use Domain\UploadRequest\Controllers\CreateFolderController;
use Domain\UploadRequest\Controllers\DeleteFileOrFolderController;
use Tests\Domain\UploadRequest\RenameFileOrFolderController; use Tests\Domain\UploadRequest\RenameFileOrFolderController;
use Domain\UploadRequest\Controllers\GetUploadRequestController; use Domain\UploadRequest\Controllers\GetUploadRequestController;
use Domain\UploadRequest\Controllers\CreateUploadRequestController; use Domain\UploadRequest\Controllers\CreateUploadRequestController;
@@ -14,6 +15,7 @@ Route::post('/{uploadRequest}/upload', UploadFilesForUploadRequestController::cl
// Edit // Edit
Route::patch('/{uploadRequest}/rename/{id}', RenameFileOrFolderController::class); Route::patch('/{uploadRequest}/rename/{id}', RenameFileOrFolderController::class);
Route::post('/{uploadRequest}/create-folder', CreateFolderController::class); Route::post('/{uploadRequest}/create-folder', CreateFolderController::class);
Route::post('/{uploadRequest}/remove', DeleteFileOrFolderController::class);
Route::group(['middleware' => ['auth:sanctum']], function () { Route::group(['middleware' => ['auth:sanctum']], function () {
Route::post('/', CreateUploadRequestController::class); Route::post('/', CreateUploadRequestController::class);

View File

@@ -65,10 +65,9 @@ class DeleteFileOrFolderAction
Storage::delete("/files/$file->user_id/$file->basename"); Storage::delete("/files/$file->user_id/$file->basename");
// Delete thumbnail if exist // Delete thumbnail if exist
if ($file->thumbnail) { if ($file->type === 'image') {
Storage::delete( getThumbnailFileList($file->basename)
"/files/$file->user_id/{$file->getRawOriginal('thumbnail')}" ->each(fn ($thumbnail) => Storage::delete("files/$file->user_id/$thumbnail"));
);
} }
// Delete file permanently // Delete file permanently
@@ -104,10 +103,9 @@ class DeleteFileOrFolderAction
Storage::delete("/files/$file->user_id/$file->basename"); Storage::delete("/files/$file->user_id/$file->basename");
// Delete thumbnail if exist // Delete thumbnail if exist
if ($file->thumbnail) { if ($file->type === 'image') {
Storage::delete( getThumbnailFileList($file->basename)
"/files/$file->user_id/{$file->getRawOriginal('thumbnail')}" ->each(fn ($thumbnail) => Storage::delete("files/$file->user_id/$thumbnail"));
);
} }
// Delete file permanently // Delete file permanently

View File

@@ -53,7 +53,7 @@ class TransferContentOwnershipToTeamFolderOwnerAction
// Move image thumbnails // Move image thumbnails
if ($file->type === 'image') { if ($file->type === 'image') {
// Get image thumbnail list // Get image thumbnail list
$thumbnailList = get_thumbnail_file_list($file->basename); $thumbnailList = getThumbnailFileList($file->basename);
// move thumbnails to the new location // move thumbnails to the new location
$thumbnailList->each(function ($basename) use ($file, $folder) { $thumbnailList->each(function ($basename) use ($file, $folder) {

View File

@@ -13,7 +13,7 @@ class DumpTrashController extends Controller
public function __invoke(): Response public function __invoke(): Response
{ {
abort_if( abort_if(
is_demo_account(auth()->user()->email), is_demo_account(),
204, 204,
'Done.' 'Done.'
); );

View File

@@ -0,0 +1,78 @@
<?php
namespace Domain\UploadRequest\Controllers;
use Domain\Files\Models\File;
use Domain\Folders\Models\Folder;
use Domain\Items\Requests\DeleteItemRequest;
use Domain\UploadRequest\Models\UploadRequest;
use Illuminate\Support\Arr;
use Illuminate\Support\Facades\Log;
use Illuminate\Support\Facades\Storage;
class DeleteFileOrFolderController
{
public function __invoke(DeleteItemRequest $request, UploadRequest $uploadRequest)
{
// Check if upload request is active
if ($uploadRequest->status !== 'active') {
return response('Gone', 410);
}
foreach ($request->input('items') as $file) {
// Get file or folder item
$item = get_item($file['type'], $file['id']);
// Delete folder
if ($file['type'] === 'folder') {
$this->destroyFolder($item);
}
// Delete file
if ($file['type'] !== 'folder') {
$this->destroyFile($item);
}
}
return response('Done', 204);
}
private function destroyFile(File $file): void
{
// Delete file
Storage::delete("/files/$file->user_id/$file->basename");
// Delete thumbnail if exist
if ($file->type === 'image') {
getThumbnailFileList($file->basename)
->each(fn ($thumbnail) => Storage::delete("files/$file->user_id/$thumbnail"));
}
// Delete file permanently
$file->forceDelete();
}
private function destroyFolder(Folder $folder): void
{
// Get children files
$files = File::whereIn('parent_id', Arr::flatten([filter_folders_ids($folder->folders), $folder->id]))
->get();
// Remove all children files
foreach ($files as $file) {
// Delete file
Storage::delete("/files/$file->user_id/$file->basename");
// Delete thumbnail if exist
if ($file->type === 'image') {
getThumbnailFileList($file->basename)
->each(fn ($thumbnail) => Storage::delete("files/$file->user_id/$thumbnail"));
}
// Delete file permanently
$file->forceDelete();
}
$folder->forceDelete();
}
}

View File

@@ -634,11 +634,11 @@ if (! function_exists('get_file_type')) {
} }
} }
if (! function_exists('get_thumbnail_file_list')) { if (! function_exists('getThumbnailFileList')) {
/** /**
* Get list of image thumbnails * Get list of image thumbnails
*/ */
function get_thumbnail_file_list(string $basename): Collection function getThumbnailFileList(string $basename): Collection
{ {
return collect([ return collect([
config('vuefilemanager.image_sizes.later'), config('vuefilemanager.image_sizes.later'),

View File

@@ -1,4 +1,5 @@
<?php <?php
namespace Tests\Domain\Files; namespace Tests\Domain\Files;
use Storage; use Storage;
@@ -47,11 +48,9 @@ class FileTest extends TestCase
'is_last' => 'true', 'is_last' => 'true',
])->assertStatus(201); ])->assertStatus(201);
$disk = Storage::disk('local');
$file = File::first(); $file = File::first();
$disk->assertMissing( Storage::assertMissing(
"chunks/$file->basename" "chunks/$file->basename"
); );
@@ -61,7 +60,7 @@ class FileTest extends TestCase
]) ])
->collapse() ->collapse()
->each( ->each(
fn ($item) => $disk->assertExists( fn($item) => Storage::assertExists(
"files/{$user->id}/{$item['name']}-{$file->basename}" "files/{$user->id}/{$item['name']}-{$file->basename}"
) )
); );
@@ -235,6 +234,63 @@ class FileTest extends TestCase
'parent_id' => $folder->id, 'parent_id' => $folder->id,
]); ]);
} }
/**
* @test
*/
public function it_delete_image_with_their_thumbnails()
{
$user = User::factory()
->hasSettings()
->create();
$image = File::factory()
->create([
'mimetype' => 'jpeg',
'type' => 'image',
'basename' => 'fake-image.jpeg',
'user_id' => $user->id,
]);
// Mock files
$thumbnail_sizes = collect([
config('vuefilemanager.image_sizes.later'),
config('vuefilemanager.image_sizes.immediately')
])->collapse();
$fakeFile = UploadedFile::fake()
->create("fake-image.jpeg", 2000, 'image/jpeg');
Storage::putFileAs("files/$user->id", $fakeFile, $fakeFile->name);
// Create fake image thumbnails
$thumbnail_sizes
->each(function ($item) use ($user) {
$fakeFile = UploadedFile::fake()
->create("{$item['name']}-fake-image.jpeg", 2000, 'image/jpeg');
Storage::putFileAs("files/$user->id", $fakeFile, $fakeFile->name);
});
$this
->actingAs($user)
->postJson('/api/remove', [
'items' => [
[
'id' => $image->id,
'type' => 'file',
'force_delete' => true,
],
],
])->assertStatus(204);
// Assert primary file was deleted
Storage::assertMissing("files/$user->id/fake-image.jpeg");
// Assert thumbnail was deleted
getThumbnailFileList("fake-image.jpeg")
->each(fn($thumbnail) => Storage::assertMissing("files/$user->id/$thumbnail"));
}
/** /**
* @test * @test

View File

@@ -2,10 +2,12 @@
namespace Tests\Domain\UploadRequest; namespace Tests\Domain\UploadRequest;
use Storage;
use Tests\TestCase; use Tests\TestCase;
use App\Users\Models\User; use App\Users\Models\User;
use Domain\Files\Models\File; use Domain\Files\Models\File;
use Domain\Folders\Models\Folder; use Domain\Folders\Models\Folder;
use Illuminate\Http\UploadedFile;
use Domain\UploadRequest\Models\UploadRequest; use Domain\UploadRequest\Models\UploadRequest;
class UploadRequestEditingTest extends TestCase class UploadRequestEditingTest extends TestCase
@@ -107,7 +109,7 @@ class UploadRequestEditingTest extends TestCase
]) ])
->assertStatus(201) ->assertStatus(201)
->assertJsonFragment([ ->assertJsonFragment([
'name' => 'New Folder', 'name' => 'New Folder',
]); ]);
$this->assertDatabaseHas('folders', [ $this->assertDatabaseHas('folders', [
@@ -115,4 +117,176 @@ class UploadRequestEditingTest extends TestCase
'parent_id' => $uploadRequest->id, 'parent_id' => $uploadRequest->id,
]); ]);
} }
/**
* @test
*/
public function it_delete_image_with_their_thumbnails()
{
$user = User::factory()
->hasSettings()
->create();
$uploadRequest = UploadRequest::factory()
->create([
'status' => 'active',
'user_id' => $user->id,
]);
$image = File::factory()
->create([
'mimetype' => 'jpeg',
'type' => 'image',
'basename' => 'fake-image.jpeg',
'user_id' => $user->id,
'parent_id' => $uploadRequest->id,
]);
// Mock files
$thumbnail_sizes = collect([
config('vuefilemanager.image_sizes.later'),
config('vuefilemanager.image_sizes.immediately')
])->collapse();
$fakeFile = UploadedFile::fake()
->create("fake-image.jpeg", 2000, 'image/jpeg');
Storage::putFileAs("files/$user->id", $fakeFile, $fakeFile->name);
// Create fake image thumbnails
$thumbnail_sizes
->each(function ($item) use ($user) {
$fakeFile = UploadedFile::fake()
->create("{$item['name']}-fake-image.jpeg", 2000, 'image/jpeg');
Storage::putFileAs("files/$user->id", $fakeFile, $fakeFile->name);
});
$this
->postJson("/api/upload-request/$uploadRequest->id/remove", [
'items' => [
[
'id' => $image->id,
'type' => 'file',
'force_delete' => true,
],
],
])->assertStatus(204);
// Assert primary file was deleted
Storage::assertMissing("files/$user->id/fake-image.jpeg");
// Assert thumbnail was deleted
getThumbnailFileList("fake-image.jpeg")
->each(fn($thumbnail) => Storage::assertMissing("files/$user->id/$thumbnail"));
}
/**
* @test
*/
public function it_delete_file()
{
$user = User::factory()
->hasSettings()
->create();
$uploadRequest = UploadRequest::factory()
->create([
'status' => 'active',
'user_id' => $user->id,
]);
$file = File::factory()
->create([
'type' => 'file',
'basename' => 'fake-file.pdf',
'user_id' => $user->id,
'parent_id' => $uploadRequest->id,
]);
$fakeFile = UploadedFile::fake()
->create('fake-file.pdf', 1200, 'application/pdf');
Storage::putFileAs("files/$user->id", $fakeFile, $fakeFile->name);
$this
->postJson("/api/upload-request/$uploadRequest->id/remove", [
'items' => [
[
'id' => $file->id,
'type' => 'file',
'force_delete' => true,
],
],
])->assertStatus(204);
// Assert primary file was deleted
Storage::assertMissing("files/$user->id/fake-file.pdf");
}
/**
* @test
*/
public function it_delete_folder_with_file_within()
{
$user = User::factory()
->hasSettings()
->create();
$uploadRequest = UploadRequest::factory()
->create([
'status' => 'active',
'user_id' => $user->id,
]);
$folder = Folder::factory()
->create([
'user_id' => $user->id,
'parent_id' => $uploadRequest->id,
]);
$folderWithin = Folder::factory()
->create([
'user_id' => $user->id,
'parent_id' => $folder->id,
]);
$file = File::factory()
->create([
'type' => 'file',
'basename' => 'fake-file.pdf',
'user_id' => $user->id,
'parent_id' => $folder->id,
]);
$fakeFile = UploadedFile::fake()
->create('fake-file.pdf', 1200, 'application/pdf');
Storage::putFileAs("files/$user->id", $fakeFile, $fakeFile->name);
$this
->postJson("/api/upload-request/$uploadRequest->id/remove", [
'items' => [
[
'id' => $folder->id,
'type' => 'folder',
'force_delete' => true,
],
],
])->assertStatus(204);
$this
->assertDatabaseMissing('folders', [
'id' => $folder->id,
])
->assertDatabaseMissing('folders', [
'id' => $folderWithin->id,
])
->assertDatabaseMissing('files', [
'id' => $file->id,
]);
// Assert primary file was deleted
Storage::assertMissing("files/$user->id/fake-file.pdf");
}
} }