From b021fa16e87587b289a875825aca82efec0f2133 Mon Sep 17 00:00:00 2001 From: Tobias Werth Date: Sat, 21 Sep 2024 06:16:20 +0200 Subject: [PATCH] Fix error handling / messaging on import validation errors. Fixes #2705. While we are at it, also improve the usefulness of the error messages. --- webapp/src/Service/ImportExportService.php | 68 +++++++++++++------ .../Unit/Service/ImportExportServiceTest.php | 12 ++-- 2 files changed, 52 insertions(+), 28 deletions(-) diff --git a/webapp/src/Service/ImportExportService.php b/webapp/src/Service/ImportExportService.php index b80dfc52cff..bfea25e3fe6 100644 --- a/webapp/src/Service/ImportExportService.php +++ b/webapp/src/Service/ImportExportService.php @@ -268,7 +268,7 @@ public function importContestData(mixed $data, ?string &$errorMessage = null, st $messages = []; /** @var ConstraintViolationInterface $error */ foreach ($errors as $error) { - $messages[] = sprintf('%s: %s', $error->getPropertyPath(), $error->getMessage()); + $messages[] = sprintf(' • `%s`: %s', $error->getPropertyPath(), $error->getMessage()); } $errorMessage = sprintf("Contest has errors:\n\n%s", implode("\n", $messages)); @@ -547,7 +547,7 @@ public function getResultsData( ); continue; } - + if (!$skip && $rank - $skippedTeams <= $contest->getGoldMedals()) { $awardString = 'Gold Medal'; $lowestMedalPoints = $teamScore->numPoints; @@ -791,10 +791,13 @@ protected function importGroupData( $messages = []; /** @var ConstraintViolationInterface $error */ foreach ($errors as $error) { - $messages[] = sprintf('%s: %s', $error->getPropertyPath(), $error->getMessage()); + $messages[] = sprintf(' • `%s`: %s', $error->getPropertyPath(), $error->getMessage()); } - $message .= sprintf("Group at index %d has errors:\n\n%s\n", $index, implode("\n", $messages)); + $message .= sprintf("Group at index %d (%s) has errors:\n%s\n\n", + $index, + json_encode($groupItem), + implode("\n", $messages)); $anyErrors = true; } else { $allCategories[] = $teamCategory; @@ -810,7 +813,7 @@ protected function importGroupData( } if ($anyErrors) { - return 0; + return -1; } foreach ($allCategories as $category) { @@ -898,10 +901,13 @@ protected function importOrganizationData( $messages = []; /** @var ConstraintViolationInterface $error */ foreach ($errors as $error) { - $messages[] = sprintf('%s: %s', $error->getPropertyPath(), $error->getMessage()); + $messages[] = sprintf(' • `%s`: %s', $error->getPropertyPath(), $error->getMessage()); } - $message .= sprintf("Organization at index %d has errors:\n\n%s\n", $index, implode("\n", $messages)); + $message .= sprintf("Organization at index %d (%s) has errors:\n%s\n\n", + $index, + json_encode($organizationItem), + implode("\n", $messages)); $anyErrors = true; } else { $allOrganizations[] = $teamAffiliation; @@ -917,7 +923,7 @@ protected function importOrganizationData( } if ($anyErrors) { - return 0; + return -1; } foreach ($allOrganizations as $organization) { @@ -1146,10 +1152,13 @@ protected function importTeamData(array $teamData, ?string &$message, ?array &$s $messages = []; /** @var ConstraintViolationInterface $error */ foreach ($errors as $error) { - $messages[] = sprintf('%s: %s', $error->getPropertyPath(), $error->getMessage()); + $messages[] = sprintf(' • `%s`: %s', $error->getPropertyPath(), $error->getMessage()); } - $message .= sprintf("Organization for team at index %d has errors:\n\n%s\n", $index, implode("\n", $messages)); + $message .= sprintf("Organization for team at index %d (%s) has errors:\n%s\n\n", + $index, + json_encode($teamItem), + implode("\n", $messages)); $anyErrors = true; } else { $createdAffiliations[] = $teamAffiliation; @@ -1169,10 +1178,13 @@ protected function importTeamData(array $teamData, ?string &$message, ?array &$s $messages = []; /** @var ConstraintViolationInterface $error */ foreach ($errors as $error) { - $messages[] = sprintf('%s: %s', $error->getPropertyPath(), $error->getMessage()); + $messages[] = sprintf(' • `%s`: %s', $error->getPropertyPath(), $error->getMessage()); } - $message .= sprintf("Organization for team at index %d has errors:\n\n%s\n", $index, implode("\n", $messages)); + $message .= sprintf("Organization for team at index %d (%s) has errors:\n%s\n\n", + $index, + json_encode($teamItem), + implode("\n", $messages)); $anyErrors = true; } else { $createdAffiliations[] = $teamAffiliation; @@ -1195,10 +1207,13 @@ protected function importTeamData(array $teamData, ?string &$message, ?array &$s $messages = []; /** @var ConstraintViolationInterface $error */ foreach ($errors as $error) { - $messages[] = sprintf('%s: %s', $error->getPropertyPath(), $error->getMessage()); + $messages[] = sprintf(' • `%s`: %s', $error->getPropertyPath(), $error->getMessage()); } - $message .= sprintf("Group for team at index %d has errors:\n\n%s\n", $index, implode("\n", $messages)); + $message .= sprintf("Group for team at index %d (%s) has errors:\n%s\n\n", + $index, + json_encode($teamItem), + implode("\n", $messages)); $anyErrors = true; } else { $createdCategories[] = $teamCategory; @@ -1244,10 +1259,13 @@ protected function importTeamData(array $teamData, ?string &$message, ?array &$s $messages = []; /** @var ConstraintViolationInterface $error */ foreach ($errors as $error) { - $messages[] = sprintf('%s: %s', $error->getPropertyPath(), $error->getMessage()); + $messages[] = sprintf(' • `%s`: %s', $error->getPropertyPath(), $error->getMessage()); } - $message .= sprintf("Team at index %d has errors:\n\n%s\n", $index, implode("\n", $messages)); + $message .= sprintf("Team at index %d (%s) has errors:\n%s\n\n", + $index, + json_encode($teamItem), + implode("\n", $messages)); $anyErrors = true; } else { $allTeams[] = $team; @@ -1265,7 +1283,7 @@ protected function importTeamData(array $teamData, ?string &$message, ?array &$s } if ($anyErrors) { - return 0; + return -1; } foreach ($createdAffiliations as $affiliation) { @@ -1348,10 +1366,13 @@ protected function importAccountData( $messages = []; /** @var ConstraintViolationInterface $error */ foreach ($errors as $error) { - $messages[] = sprintf('%s: %s', $error->getPropertyPath(), $error->getMessage()); + $messages[] = sprintf(' • `%s`: %s', $error->getPropertyPath(), $error->getMessage()); } - $message .= sprintf("Team for user at index %d has errors:\n\n%s\n", $index, implode("\n", $messages)); + $message .= sprintf("Team for user at index %d (%s) has errors:\n%s\n\n", + $index, + json_encode($accountItem), + implode("\n", $messages)); $anyErrors = true; } else { $newTeams[] = [ @@ -1397,10 +1418,13 @@ protected function importAccountData( $messages = []; /** @var ConstraintViolationInterface $error */ foreach ($errors as $error) { - $messages[] = sprintf('%s: %s', $error->getPropertyPath(), $error->getMessage()); + $messages[] = sprintf(' • `%s`: %s', $error->getPropertyPath(), $error->getMessage()); } - $message .= sprintf("User at index %d has errors:\n\n%s\n", $index, implode("\n", $messages)); + $message .= sprintf("User at index %d (%s) has errors:\n%s\n\n", + $index, + json_encode($accountItem), + implode("\n", $messages)); $anyErrors = true; } else { $allUsers[] = $user; @@ -1412,7 +1436,7 @@ protected function importAccountData( } if ($anyErrors) { - return 0; + return -1; } foreach ($allUsers as $user) { diff --git a/webapp/tests/Unit/Service/ImportExportServiceTest.php b/webapp/tests/Unit/Service/ImportExportServiceTest.php index 1808ccb316e..8f77fe0a05b 100644 --- a/webapp/tests/Unit/Service/ImportExportServiceTest.php +++ b/webapp/tests/Unit/Service/ImportExportServiceTest.php @@ -105,7 +105,7 @@ public function provideImportContestDataErrors(): Generator 'start-time' => '2020-01-01T12:34:56+02:00', 'scoreboard-freeze-length' => '30:00', ], - "Contest has errors:\n\nname: This value should not be blank.\nshortname: This value should not be blank." + "Contest has errors:\n\n • `name`: This value should not be blank.\n • `shortname`: This value should not be blank." ]; } @@ -501,7 +501,7 @@ public function testImportAccountsJsonError(): void // Remove the file, we don't need it anymore. unlink($fileName); - self::assertEquals(0, $importCount); + self::assertEquals(-1, $importCount); self::assertMatchesRegularExpression('/Only alphanumeric characters and _-@. are allowed/', $message); $postCount = $em->getRepository(User::class)->count([]); @@ -870,7 +870,7 @@ public function testImportTeamsJsonError(): void // Remove the file, we don't need it anymore. unlink($fileName); - self::assertMatchesRegularExpression('/name: This value should not be blank./', $message); + self::assertMatchesRegularExpression('/.*name: This value should not be blank.*/', $message); self::assertEquals(0, $importCount); $postCount = $em->getRepository(Team::class)->count([]); @@ -908,7 +908,7 @@ public function testImportTeamsJsonErrorEmptyString(): void // Remove the file, we don't need it anymore. unlink($fileName); - self::assertMatchesRegularExpression('/name: This value should not be blank./', $message); + self::assertMatchesRegularExpression('/.*name: This value should not be blank.*/', $message); self::assertEquals(0, $importCount); $postCount = $em->getRepository(Team::class)->count([]); @@ -1050,7 +1050,7 @@ public function testImportGroupsJsonError(): void // Remove the file, we don't need it anymore. unlink($fileName); - self::assertMatchesRegularExpression('/name: This value should not be blank/', $message); + self::assertMatchesRegularExpression('/.*name: This value should not be blank.*/', $message); self::assertEquals(0, $importCount); $postCount = $em->getRepository(TeamCategory::class)->count([]); @@ -1150,7 +1150,7 @@ public function testImportOrganizationsErrorJson(): void unlink($fileName); self::assertMatchesRegularExpression('/ISO3166-1 alpha-3 values are allowed/', $message); - self::assertEquals(0, $importCount); + self::assertEquals(-1, $importCount); $postCount = $em->getRepository(TeamAffiliation::class)->count([]); self::assertEquals($preCount, $postCount);