From 2bac3e5e1223de46b5bcd9aa2fc0cf6b71f9c687 Mon Sep 17 00:00:00 2001
From: Tobias Gross-Vogt <tgros@tugraz.at>
Date: Tue, 21 Jun 2022 11:43:52 +0200
Subject: [PATCH] pagination update

---
 composer.lock                                 | 18 ++++-------------
 src/Controller/GetAttendeesByCourse.php       | 20 +++++++++----------
 .../CourseCollectionDataProvider.php          | 16 +++++++--------
 src/Entity/Course.php                         |  8 +++++++-
 src/Entity/CourseAttendee.php                 |  4 +++-
 5 files changed, 30 insertions(+), 36 deletions(-)

diff --git a/composer.lock b/composer.lock
index 2e1a278..a012231 100644
--- a/composer.lock
+++ b/composer.lock
@@ -172,14 +172,8 @@
             "version": "dev-main",
             "source": {
                 "type": "git",
-                "url": "https://gitlab.tugraz.at/dbp/relay/dbp-relay-core-bundle.git",
-                "reference": "766488dd1db1778632dedd0752f89e065c68a2da"
-            },
-            "dist": {
-                "type": "zip",
-                "url": "https://gitlab.tugraz.at/api/v4/projects/dbp%2Frelay%2Fdbp-relay-core-bundle/repository/archive.zip?sha=766488dd1db1778632dedd0752f89e065c68a2da",
-                "reference": "766488dd1db1778632dedd0752f89e065c68a2da",
-                "shasum": ""
+                "url": "git@gitlab.tugraz.at:dbp/relay/dbp-relay-core-bundle.git",
+                "reference": "c7bab5379711297e01e7d53c231d295d732e1d45"
             },
             "require": {
                 "api-platform/core": "^2.6.6",
@@ -275,11 +269,7 @@
                 "AGPL-3.0-or-later"
             ],
             "description": "The core bundle of the Relay API gateway",
-            "support": {
-                "source": "https://gitlab.tugraz.at/dbp/relay/dbp-relay-core-bundle/-/tree/main",
-                "issues": "https://gitlab.tugraz.at/dbp/relay/dbp-relay-core-bundle/-/issues"
-            },
-            "time": "2022-06-11T02:25:45+02:00"
+            "time": "2022-06-21T09:42:10+00:00"
         },
         {
             "name": "doctrine/annotations",
@@ -10064,5 +10054,5 @@
     "platform-overrides": {
         "php": "7.3"
     },
-    "plugin-api-version": "2.3.0"
+    "plugin-api-version": "2.2.0"
 }
diff --git a/src/Controller/GetAttendeesByCourse.php b/src/Controller/GetAttendeesByCourse.php
index c55166e..ef10d2a 100644
--- a/src/Controller/GetAttendeesByCourse.php
+++ b/src/Controller/GetAttendeesByCourse.php
@@ -5,34 +5,32 @@ declare(strict_types=1);
 namespace Dbp\Relay\BaseCourseBundle\Controller;
 
 use Dbp\Relay\BaseCourseBundle\API\CourseProviderInterface;
-use Dbp\Relay\CoreBundle\Helpers\ArrayFullPaginator;
+use Dbp\Relay\CoreBundle\Pagination\Pagination;
+use Dbp\Relay\CoreBundle\Pagination\Paginator;
 use Symfony\Bundle\FrameworkBundle\Controller\AbstractController;
 use Symfony\Component\HttpFoundation\Request;
 
 class GetAttendeesByCourse extends AbstractController
 {
-    public const ITEMS_PER_PAGE = 250;
+    public const MAX_ITEMS_PER_PAGE = 250;
 
-    protected $coursesProvider;
+    /** @var CourseProviderInterface */
+    private $coursesProvider;
 
     public function __construct(CourseProviderInterface $coursesProvider)
     {
         $this->coursesProvider = $coursesProvider;
     }
 
-    public function __invoke(string $identifier, Request $request): ArrayFullPaginator
+    public function __invoke(string $identifier, Request $request): Paginator
     {
         $this->denyAccessUnlessGranted('IS_AUTHENTICATED_FULLY');
 
-        $page = (int) $request->query->get('page', 1);
-        $perPage = (int) $request->query->get('perPage', self::ITEMS_PER_PAGE);
-
         $options = [];
-        $lang = $request->query->get('lang', 'de');
-        $options['lang'] = $lang;
+        $options['lang'] = $request->query->get('lang', 'de');
 
-        $courses = $this->coursesProvider->getAttendeesByCourse($identifier, $options);
+        Pagination::addPaginationOptions($options, $request->query->all(), self::MAX_ITEMS_PER_PAGE);
 
-        return new ArrayFullPaginator($courses, $page, $perPage);
+        return $this->coursesProvider->getAttendeesByCourse($identifier, $options);
     }
 }
diff --git a/src/DataProvider/CourseCollectionDataProvider.php b/src/DataProvider/CourseCollectionDataProvider.php
index b44a363..621e291 100644
--- a/src/DataProvider/CourseCollectionDataProvider.php
+++ b/src/DataProvider/CourseCollectionDataProvider.php
@@ -17,13 +17,9 @@ final class CourseCollectionDataProvider extends AbstractController implements C
     /** @var CourseProviderInterface */
     private $courseProvider;
 
-    /** @var Pagination */
-    private $pagination;
-
-    public function __construct(CourseProviderInterface $courseProvider, Pagination $pagination)
+    public function __construct(CourseProviderInterface $courseProvider)
     {
         $this->courseProvider = $courseProvider;
-        $this->pagination = $pagination;
     }
 
     public function supports(string $resourceClass, string $operationName = null, array $context = []): bool
@@ -36,7 +32,6 @@ final class CourseCollectionDataProvider extends AbstractController implements C
         $this->denyAccessUnlessGranted('IS_AUTHENTICATED_FULLY');
 
         $filters = $context['filters'] ?? [];
-
         $options = [];
         $options['lang'] = $filters['lang'] ?? 'de';
 
@@ -58,9 +53,11 @@ final class CourseCollectionDataProvider extends AbstractController implements C
         $filterByOrganizationId = $organizationId !== '';
         $filterByLecturerId = $lecturerId !== '';
 
+        dump($context);
+
         if (!($filterByOrganizationId && $filterByLecturerId)) {
-            $this->pagination->addPaginationOptions($options, $resourceClass, $operationName, $context);
-        } // else -> request the whole set of results
+            Pagination::addPaginationOptions($options, $filters);
+        } // else (both filters provided) -> request paginators holding the whole set of results since we need to intersect them
 
         $coursePaginator = null;
         if ($filterByOrganizationId || $filterByLecturerId) {
@@ -76,7 +73,8 @@ final class CourseCollectionDataProvider extends AbstractController implements C
                     $intersection = array_uintersect($coursePaginator->getItems(), $coursesByPersonPaginator->getItems(),
                         'Dbp\Relay\BaseCourseBundle\DataProvider\CourseCollectionDataProvider::compareCourses');
                     $courses = array_values($intersection);
-                    $coursePaginator = $this->pagination->createWholeResultPaginator($courses, $resourceClass, $operationName, $context);
+                    Pagination::addPaginationOptions($options, $filters);
+                    $coursePaginator = Pagination::createWholeResultPaginator($courses, $options);
                 }
             }
         } else {
diff --git a/src/Entity/Course.php b/src/Entity/Course.php
index 25147fa..adfdc4a 100644
--- a/src/Entity/Course.php
+++ b/src/Entity/Course.php
@@ -13,6 +13,8 @@ use Dbp\Relay\CoreBundle\LocalData\LocalDataAwareTrait;
  *     collectionOperations={
  *         "get" = {
  *             "path" = "/base/courses",
+ *             "security" = "is_granted('IS_AUTHENTICATED_FULLY')",
+ *             "pagination_client_partial" = true,
  *             "openapi_context" = {
  *                 "tags" = {"BaseCourse"},
  *                 "parameters" = {
@@ -21,7 +23,8 @@ use Dbp\Relay\CoreBundle\LocalData\LocalDataAwareTrait;
  *                     {"name" = "term", "in" = "query", "description" = "Teaching term", "type" = "string", "enum" = {"W", "S"}, "example" = "W"},
  *                     {"name" = "organization", "in" = "query", "description" = "Get courses of an organization (ID of BaseOrganization resource)", "required" = false, "type" = "string", "example" = "1190"},
  *                     {"name" = "lecturer", "in" = "query", "description" = "Get courses of a lecturer (ID of BasePerson resource)", "required" = false, "type" = "string", "example" = "woody007"},
- *                     {"name" = "includeLocal", "in" = "query", "description" = "Local data attributes to include", "type" = "string", "example" = "BaseCourse.code,BaseCourse.numberOfCredits"}
+ *                     {"name" = "includeLocal", "in" = "query", "description" = "Local data attributes to include", "type" = "string", "example" = "BaseCourse.code,BaseCourse.numberOfCredits"},
+ *                     {"name" = "partialPagination", "in" = "query", "description" = "Enable partial pagination", "type" = "bool", "example" = "false"}
  *                 }
  *             }
  *         }
@@ -29,6 +32,7 @@ use Dbp\Relay\CoreBundle\LocalData\LocalDataAwareTrait;
  *     itemOperations={
  *         "get" = {
  *             "path" = "/base/courses/{identifier}",
+ *             "security" = "is_granted('IS_AUTHENTICATED_FULLY')",
  *             "openapi_context" = {
  *                 "tags" = {"BaseCourse"},
  *                 "parameters" = {
@@ -48,4 +52,6 @@ class Course implements CourseInterface, LocalDataAwareInterface
 {
     use LocalDataAwareTrait;
     use CourseTrait;
+
+    public const SEARCH_PARAMETER_NAME = 'search';
 }
diff --git a/src/Entity/CourseAttendee.php b/src/Entity/CourseAttendee.php
index c8ed97b..aa6f2f7 100644
--- a/src/Entity/CourseAttendee.php
+++ b/src/Entity/CourseAttendee.php
@@ -23,6 +23,7 @@ use Symfony\Component\Serializer\Annotation\Groups;
  *             "method" = "GET",
  *             "path" = "/base/courses/{identifier}/attendees",
  *             "security" = "is_granted('IS_AUTHENTICATED_FULLY')",
+ *             "pagination_client_partial" = true,
  *             "controller" = GetAttendeesByCourse::class,
  *             "read" = false,
  *             "normalization_context" = {
@@ -34,7 +35,8 @@ use Symfony\Component\Serializer\Annotation\Groups;
  *                 "summary" = "Get the attendees of a course.",
  *                 "parameters" = {
  *                     {"name" = "identifier", "in" = "path", "description" = "Resource identifier", "required" = true, "type" = "string", "example" = "257571"},
- *                     {"name" = "lang", "in" = "query", "description" = "Language of result", "type" = "string", "enum" = {"de", "en"}, "example" = "de"}
+ *                     {"name" = "lang", "in" = "query", "description" = "Language of result", "type" = "string", "enum" = {"de", "en"}, "example" = "de"},
+ *                     {"name" = "partial", "in" = "query", "description" = "Enable partial pagination", "type" = "bool", "example" = "false"}
  *                 }
  *             }
  *         }
-- 
GitLab