Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Model::paginate() behavior when $page exceeds the last page #8904

Open
crustamet opened this issue May 19, 2024 · 21 comments
Open

Model::paginate() behavior when $page exceeds the last page #8904

crustamet opened this issue May 19, 2024 · 21 comments
Labels
enhancement PRs that improve existing functionalities

Comments

@crustamet
Copy link
Contributor

crustamet commented May 19, 2024

PHP Version

8.3

CodeIgniter4 Version

4.1.5

CodeIgniter4 Installation Method

Manual (zip or tar.gz)

Which operating systems have you tested for this bug?

Linux

Which server did you use?

apache

Database

MySQL 5.6

What happened?

When i paginate the model like this $UsersMode->orderBy('id_user', 'ASC')->paginate(8, 'group', $page_number);

returns "SELECT * FROM users ORDER BY id_user ASC LIMIT 8, 8" based on offset and per page number

But my total count rows in the table are 24, so when i reach the end 16, 8 it returns me the same query as the last existing page in my database even with $page_number = 100

Steps to Reproduce

	public function load_more($page_number = 2)
	{
		$UsersModel 	= new UsersModel();
		
		$this->ViewData['Page'] = $page_number;
		
		$this->ViewData['Users'] = $UsersMode->orderBy('id_user', 'ASC')->paginate(8, 'group', $page_number);

		echo $UsersModel->db->getLastQuery();
	}

returns "SELECT * FROM users ORDER BY id_user ASC LIMIT 8, 8"

But my total count rows in the table are 24, so when i reach the end 16, 8 $page_number = 10 and go to a page number more than 3 it returns the same query "SELECT * FROM users ORDER BY id_user ASC LIMIT 16, 8"

Expected Output

Going to a page number more than the total count rows of my table it should return the propper offset even if i have the total rows less than the per page number or less than the total count

$page_number = 10 it should returns the query "SELECT * FROM users ORDER BY id_user ASC LIMIT 80, 8" even if i don't have enough records in my database table

Anything else?

No response

@crustamet crustamet added the bug Verified issues on the current code behavior or pull requests that will fix them label May 19, 2024
@crustamet
Copy link
Contributor Author

crustamet commented May 19, 2024

one this BaseModel
https://github.com/codeigniter4/CodeIgniter4/blob/develop/system/BaseModel.php#L1273

	public function paginate(?int $perPage = null, string $group = 'default', ?int $page = null, int $segment = 0)
	{
		// Since multiple models may use the Pager, the Pager must be shared.
		$pager = service('pager');

		if ($segment !== 0) {
			$pager->setSegment($segment, $group);
		}

		$page = $page >= 1 ? $page : $pager->getCurrentPage($group);
		// Store it in the Pager library, so it can be paginated in the views.
		$this->pager = $pager->store($group, $page, $perPage, $this->countAllResults(false), $segment);
		$perPage     = $this->pager->getPerPage($group);
		$offset      = ($pager->getCurrentPage($group) - 1) * $perPage;

		return $this->findAll($perPage, $offset);
	}

should be converted to something like this

	public function paginate(?int $perPage = null, string $group = 'default', ?int $page = null, int $segment = 0)
	{
		// Since multiple models may use the Pager, the Pager must be shared.
		$pager = service('pager');

		if ($segment !== 0) {
			$pager->setSegment($segment, $group);
		}

		$page = $page >= 1 ? $page : $pager->getCurrentPage($group);
		// Store it in the Pager library, so it can be paginated in the views.
		$this->pager = $pager->store($group, $page, $perPage, $this->countAllResults(false), $segment);
		$offset      = ($page) * $perPage;
		$perPage     = $this->pager->getPerPage($group);

		return $this->findAll($perPage, $offset);
	}

@kenjis
Copy link
Member

kenjis commented May 19, 2024

@crustamet I don't understand the difference. Please show the diff by diff command or git diff.

@kenjis
Copy link
Member

kenjis commented May 19, 2024

$page_number = 10 it should returns the query "SELECT * FROM users ORDER BY id_user ASC LIMIT 80, 8" even if i don't have enough records in my database table

Why?

@crustamet
Copy link
Contributor Author

crustamet commented May 19, 2024

it doesn't matter the difference because how i did it is no good it is just to point out that if the total records in the database are 10

when you go to page 1000

the query remains as of the last page so the limit whould be {a long number}, {8}

I dont have pagination only a load more button

"Why?" - because i cannot remove the button when i have no records, it always gives me records even when i don't have so it always shows load more

@crustamet
Copy link
Contributor Author

crustamet commented May 20, 2024

$page_number = 10 it should returns the query "SELECT * FROM users ORDER BY id_user ASC LIMIT 80, 8" even if i don't have enough records in my database table

Why?

First of all it should return that query if you use

$Model->paginate(8, 'group', 10);
it should create the query above ? right or wrong ?

While the current implemented code is dependent on how many rows you actually have in the table

If you have 10 users and you are at page 10

$Model->paginate(8, 'group', 10);
This returns "SELECT * FROM users ORDER BY id_user ASC LIMIT 16, 8"

I cannot return a 404 page if i have no more users on page 10 LOL

@kenjis kenjis removed the bug Verified issues on the current code behavior or pull requests that will fix them label May 20, 2024
@kenjis
Copy link
Member

kenjis commented May 20, 2024

It seems this behavior is intentional.

public function testPaginatePageOutOfRange(): void
{
$this->createModel(ValidModel::class);
$this->model->paginate(1, 'default', -500);
$this->assertSame(1, $this->model->pager->getCurrentPage());
$this->model->paginate(1, 'default', 500);
$this->assertSame($this->model->pager->getPageCount(), $this->model->pager->getCurrentPage());
}

@kenjis
Copy link
Member

kenjis commented May 20, 2024

You can check if it has any more pages with $pager->hasMore().

@crustamet
Copy link
Contributor Author

Ok i get what you are saying, I dig this $pager->hasMore() method it does what I need a kind of a checker, but i have millions in my database, i just can't wrap my head around this one why it is an intentional behavior. while the query speed is slower when you select from millions at the last page than if you select from millions from a page that does not exists.
Furthermore you don't actually need to add additional code to check this as pagination model will return an empty array if you go out of range

Anyways up to you guys, $pager->hasMore() is missing from the documentation

@kenjis
Copy link
Member

kenjis commented May 20, 2024

I don't know why the pagination behaves like that.
But test code is document that describes how the system works,
so as you see the test code, the current behavior is intentional.

If we change the behavior, it will be an breaking change.

If you insist that this behavior is something that needs to be corrected,
please send a PR to 4.6 branch and explain reasons that many people would agree.
If it is to meet the needs of just one of you, you can simply customize the pagination.

In my opinion, this pagination was not originally designed to handle very large amounts of data at high speed.
If you want to process large amounts of data at high speed,
I recommend a different implementation without using OFFSET.

@crustamet
Copy link
Contributor Author

I agree with everything you said, actually i will extend the current BaseModel paginate() maybe in time i will understand why it returns data even if it is out of range.

I hope some other guy read this and maybe help us understand better

@kenjis kenjis changed the title Bug: Paginate on model the pageNumber with offset does not work properly Model::paginate() behavior when $page exceeds the last page Jun 6, 2024
@neznaika0
Copy link
Contributor

I've known about this behavior for a long time.
At first, this caused a misunderstanding of "Why?" Such a feature, I think, was popular in 2009.
This is not critical at this stage. Problems may appear if you use paginate() for the API (strict result is important).
As a fix, it is possible to create a new strictPaginate() method and remove this behavior

@kenjis
Copy link
Member

kenjis commented Jun 7, 2024

The method name pagenateStrict() is better, but I agree with adding the new method.

@crustamet
Copy link
Contributor Author

crustamet commented Jun 28, 2024

i made it and tested it
The old one

    public function paginate(?int $perPage = null, string $group = 'default', ?int $page = null, int $segment = 0)
    {
        // Since multiple models may use the Pager, the Pager must be shared.
        $pager = Services::pager();

        if ($segment) {
            $pager->setSegment($segment, $group);
        }

        $page = $page >= 1 ? $page : $pager->getCurrentPage($group);
        // Store it in the Pager library, so it can be paginated in the views.
        $this->pager = $pager->store($group, $page, $perPage, $this->countAllResults(false), $segment);
        $perPage     = $this->pager->getPerPage($group);
        $offset      = ($pager->getCurrentPage($group) - 1) * $perPage;

        return $this->findAll($perPage, $offset);
    }

The new one

	public function paginateStrict(?int $perPage = null, string $group = 'default', ?int $page = null, int $segment = 0)
	{
		$pager = Services::pager();
		
		if ($segment) {
			$pager->setSegment($segment, $group);
		}
	
		$page = $page >= 1 ? $page : $pager->getCurrentPage($group);
		
		// Store it in the Pager library, so it can be paginated in the views.
		$this->pager = $pager->store($group, $page, $perPage, $this->countAllResults(false), $segment);
		
		$perPage     = $this->pager->getPerPage($group);
		$offset      = ($page - 1) * $perPage;
	
		return $this->findAll($perPage, $offset);
	}

@kenjis kenjis added the enhancement PRs that improve existing functionalities label Jul 1, 2024
@neznaika0
Copy link
Contributor

I've looked at the code.

In a good way, you need to change Pager->store(), $page is illogically set in it, $page has a "raw" value in line 148, and then it in line 154 is changed to limit.

if ($segment > 0 && $this->groups[$group]['currentPage'] > 0) {
$page = $this->groups[$group]['currentPage'];
}
$perPage ??= $this->config->perPage;
$pageCount = (int) ceil($total / $perPage);
$this->groups[$group]['currentPage'] = $page > $pageCount ? $pageCount : $page;

Similar actions in BaseModel->paginate(): On line 1282 it has one value, then it is not used for $offset

$page = $page >= 1 ? $page : $pager->getCurrentPage($group);
// Store it in the Pager library, so it can be paginated in the views.
$this->pager = $pager->store($group, $page, $perPage, $this->countAllResults(false), $segment);
$perPage = $this->pager->getPerPage($group);
$offset = ($pager->getCurrentPage($group) - 1) * $perPage;

So the main reason is to change the currentPage.
Will we find more options instead of editing the BaseModel?

@crustamet
Copy link
Contributor Author

I think this feature was implemented in the Feature.php

/**
 * The behavior of `limit(0)` in Query Builder.
 *
 * If true, `limit(0)` returns all records. (the behavior of 4.4.x or before in version 4.x.)
 * If false, `limit(0)` returns no records. (the behavior of 3.1.9 or later in version 3.x.)
 */
public bool $limitZeroAsAll = false;

Right ? this issue should be closed.

@neznaika0
Copy link
Contributor

No. limit(0) ‐ switches receiving 1000/1000 rows or 0/1000 rows

@crustamet
Copy link
Contributor Author

So with with this

public bool $limitZeroAsAll = false;

If I have a total of 10 Pages
If I go to the Page 11, will get records ? or not ? :)))

If you still get records for sure this is not implemented... and in fact this limitZeroAsAll has nothing to do with pagination

@neznaika0
Copy link
Contributor

#8278

@crustamet
Copy link
Contributor Author

crustamet commented Jan 10, 2025

i don't think you got my issue, This is another issue #8278

Please answer my question as the answer matters the most

If I have a total of 10 Pages
If I go to the Page 11, will get records ? or not ?

And it has nothing to do with ->limit(0,0);

I've tested it the answer is still YES they are still returning data even tough you are exceeding the total pages

If i go to the Page 11, but I only have records for 10 pages it should return NO RECORDS !

Why ? because of SEO they see duplicate pages and someone can create issues by just adding +1 to the page number

Page 11, Page 12, Page 10000 they will be all issues ! of duplicate content

@neznaika0
Copy link
Contributor

I understood the problem, but you don't understand the $limitZeroAsAll parameter. That's why I pointed to the issue. The answer to your question is: it doesn't affect Pager.

You can extend the Pager class and overwrite the service. All. This is the fastest solution for you right now.
https://codeigniter4.github.io/userguide/extending/core_classes.html#replacing-core-classes

@crustamet
Copy link
Contributor Author

I know @neznaika0 i did extended the pager for sure as i did not have any other solution.

that parameter $limitZeroAsAll should be related with the Pager from my point of view...

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement PRs that improve existing functionalities
Projects
None yet
Development

No branches or pull requests

3 participants