Skip to content

Commit

Permalink
#6528 Allow bulk deletion of Incomplete Submissions
Browse files Browse the repository at this point in the history
  • Loading branch information
taslangraham authored Dec 13, 2024
2 parents 88f7c6e + 6fe672b commit 2837b80
Show file tree
Hide file tree
Showing 5 changed files with 119 additions and 2 deletions.
83 changes: 83 additions & 0 deletions api/v1/_submissions/PKPBackendSubmissionsController.php
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@
use APP\core\Application;
use APP\facades\Repo;
use APP\submission\Collector;
use APP\submission\Submission;
use Illuminate\Http\JsonResponse;
use Illuminate\Http\Request;
use Illuminate\Http\Response;
Expand Down Expand Up @@ -87,6 +88,16 @@ public function getGroupRoutes(): void
),
]);

Route::delete('', $this->bulkDeleteIncompleteSubmissions(...))
->name('_submission.incomplete.delete')
->middleware([
self::roleAuthorizer([
Role::ROLE_ID_SITE_ADMIN,
Role::ROLE_ID_MANAGER,
Role::ROLE_ID_AUTHOR,
]),
]);

Route::delete('{submissionId}', $this->delete(...))
->name('_submission.delete')
->middleware([
Expand Down Expand Up @@ -430,6 +441,78 @@ public function delete(Request $illuminateRequest): JsonResponse
return response()->json([], Response::HTTP_OK);
}

/**
* Delete a list of incomplete submissions
*/
public function bulkDeleteIncompleteSubmissions(Request $illuminateRequest): JsonResponse
{
$submissionIdsRaw = paramToArray($illuminateRequest->query('ids') ?? []);

if (empty($submissionIdsRaw)) {
return response()->json([
'error' => __('api.submission.400.missingQueryParam'),
], Response::HTTP_BAD_REQUEST);
}

$submissionIds = [];

foreach ($submissionIdsRaw as $id) {
$integerId = intval($id);

if (!$integerId) {
return response()->json([
'error' => __('api.submission.400.invalidId', ['id' => $id])
], Response::HTTP_BAD_REQUEST);
}

$submissionIds[] = $id;
}

$collector = $this->getSubmissionCollector($illuminateRequest->query())
->filterBySubmissionIds($submissionIds)
->filterByIncomplete(true);

$request = Application::get()->getRequest();
$context = $this->getRequest()->getContext();
$user = $request->getUser();

if ($user->hasRole([Role::ROLE_ID_AUTHOR], $context->getId())) {
$userId = $request->getUser()->getId();
$collector->assignedTo([$userId]);
}

$submissions = $collector->getMany()->all();

$submissionIdsFound = array_map(fn (Submission $submission) => $submission->getData('id'), $submissions);

if (array_diff($submissionIds, $submissionIdsFound)) {
return response()->json([
'error' => __('api.404.resourceNotFound')
], Response::HTTP_NOT_FOUND);
}


foreach ($submissions as $submission) {
if ($context->getId() != $submission->getData('contextId')) {
return response()->json([
'error' => __('api.submissions.403.deleteSubmissionOutOfContext'),
], Response::HTTP_FORBIDDEN);
}

if (!Repo::submission()->canCurrentUserDelete($submission)) {
return response()->json([
'error' => __('api.submissions.403.unauthorizedDeleteSubmission'),
], Response::HTTP_FORBIDDEN);
}
}

foreach ($submissions as $submission) {
Repo::submission()->delete($submission);
}

return response()->json([], Response::HTTP_OK);
}

/**
* Configure a submission Collector based on the query params
*/
Expand Down
16 changes: 16 additions & 0 deletions classes/submission/Collector.php
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@ abstract class Collector implements CollectorInterface, ViewsCount
public DAO $dao;
public ?array $categoryIds = null;
public ?array $contextIds = null;
public ?array $submissionIds = null;
public ?int $count = null;
public ?int $daysInactive = null;
public bool $isIncomplete = false;
Expand Down Expand Up @@ -252,6 +253,17 @@ public function filterByRevisionsSubmitted(?bool $revisionsSubmitted): AppCollec
return $this;
}

/**
* Limit results to only submissions with the specified IDs
*
* @param ?int[] $submissionIds Submission IDs
*/
public function filterBySubmissionIds(?array $submissionIds): static
{
$this->submissionIds = $submissionIds;
return $this;
}

/**
* Limit results to submissions assigned to these users
*
Expand Down Expand Up @@ -372,6 +384,10 @@ public function getQueryBuilder(): Builder
$q->whereIn('s.context_id', $this->contextIds);
}

if (isset($this->submissionIds)) {
$q->whereIn('s.submission_id', array_map(intval(...), $this->submissionIds));
}

// Prepare keywords (allows short and numeric words)
$keywords = collect(Application::getSubmissionSearchIndex()->filterKeywords($this->searchPhrase, false, true, true))
->unique()
Expand Down
6 changes: 6 additions & 0 deletions locale/en/api.po
Original file line number Diff line number Diff line change
Expand Up @@ -347,5 +347,11 @@ msgstr "Only 'accept' and 'decline' are valid values"
msgid "api.submission.400.sectionDoesNotExist"
msgstr "The provided section does not exist."

msgid "api.submission.400.missingQueryParam"
msgstr "The request is missing the required query parameter `ids`. Please provide the `ids` of the submissions you wish to delete."

msgid "api.submission.400.invalidId"
msgstr "Invalid ID: \"{$id}\" provided."

msgid "api.publications.403.noEnabledIdentifiers"
msgstr "Publication identifiers form is unavailable as there are no enabled Identifiers."
2 changes: 1 addition & 1 deletion locale/en/common.po
Original file line number Diff line number Diff line change
Expand Up @@ -806,7 +806,7 @@ msgid "common.replaceFile"
msgstr "Replace file"

msgid "common.requiredField"
msgstr "Required fields are marked with an asterisk: <abbr class="required" title="required">*</abbr>"
msgstr "Required fields are marked with an asterisk: <abbr class=\"required\" title=\"required\">*</abbr>"

msgid "common.required"
msgstr "Required"
Expand Down
14 changes: 13 additions & 1 deletion locale/en/submission.po
Original file line number Diff line number Diff line change
Expand Up @@ -2633,4 +2633,16 @@ msgid "fileManager.productionReadyFilesDescription"
msgstr "These are the files that will be sent for publication"

msgid "reviewerManager.reviewerStatus"
msgstr "Reviewer status"
msgstr "Reviewer status"

msgid "dashboard.submissions.incomplete.bulkDelete.confirm"
msgstr "Confirm Delete of Incomplete Submissions"

msgid "dashboard.submissions.incomplete.bulkDelete.body"
msgstr "Are you sure you want to delete the selected items? This action cannot be undone. Please confirm to proceed."

msgid "dashboard.submissions.incomplete.bulkDelete.column.description"
msgstr "Select incomplete submissions to be deleted."

msgid "dashboard.submissions.incomplete.bulkDelete.button"
msgstr "Delete Incomplete Submissions"

0 comments on commit 2837b80

Please sign in to comment.