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

Data providers perform unnecessary COUNT queries that negatively affect performance #20213

Closed
sleptor opened this issue Jun 25, 2024 · 18 comments
Milestone

Comments

@sleptor
Copy link

sleptor commented Jun 25, 2024

What steps will reproduce the problem?

Check the debug panel on any page containing GridView.

What is the expected result?

Single COUNT query

What do you get instead?

Multiple identical COUNT queries

Additional info

Q A
Yii version 2.0.51-dev
PHP version 8.1
Operating system Debian Buster

The problem is in this function:
https://github.com/yiisoft/yii2/blob/master/framework/data/BaseDataProvider.php#L165 and related to the changes made in this commit lav45@bcc499b

It is expected to implement caching, but due to a bug/typo it always calls $this->prepareTotalCount().

The correct code has to be the following:

  public function getTotalCount()
    {
        if ($this->_pagination === false) {
            return $this->getCount();
        }
        if ($this->_totalCount !== null) {
            return (int)$this->_totalCount;
        }
        return $this->_totalCount = $this->prepareTotalCount();  // <---
    }
@lav45
Copy link
Contributor

lav45 commented Jun 25, 2024

@sleptor

So, the cache definitely does not work, which affects performance notably.

Usually getTotalCount() is called once when creating a paginated navigation.
If several identical queries are running in your case, then use caching.

User::getDb()->queryCache = new \yii\caching\ArrayCache();

@sleptor
Copy link
Author

sleptor commented Jun 25, 2024

@lav45 getTotalCount() is being called multiple times during GridView rendering. It's normal.
To avoid duplicated COUNT queries, its result is cached in $this->_totalCount.

However, due to the bug, $this->_totalCount is never initialized, so it's always null.

@sleptor
Copy link
Author

sleptor commented Jun 25, 2024

@lav45, it's regression. In 2.0.49, the caching logic is correct. $this->_totalCount is initialized when it's null

   public function getTotalCount()
    {
        if ($this->getPagination() === false) {
            return $this->getCount();
        } elseif ($this->_totalCount === null) {
            $this->_totalCount = $this->prepareTotalCount();
        }

        return $this->_totalCount;
    }

@lav45
Copy link
Contributor

lav45 commented Jun 25, 2024

@rob006
Copy link
Contributor

rob006 commented Jun 25, 2024

@lav45 This is a new test, which is not even released and changes behavior that existed in framework since the beginning (and probably breaks BC in some cases) - I think it is fine to drop it if it creates any problems.

In general problems started with #20070 (preceded by #20007) and each attempt to fix them created even more problems and regressions. I suggest to revert all changes to state before #20070 or #20007 and start again from there. Messing with pagination and data provider cache looks like a dead end to me.

@sleptor
Copy link
Author

sleptor commented Jun 25, 2024

@lav45 You added this test, and I think it's wrong because it's artificial. What would a real-life situation be when you need to change the provider's query after a pagination call?

2.0.49 can't pass this test also.

It breaks BC. It ruins performance.

I support the idea of reverting all your changes.

@marcovtwout
Copy link
Member

marcovtwout commented Jul 9, 2024

What is needed to solve the immediate issue here? Can we revert all the mensioned changes so we can release 2.0.50.1? 2.0.50 is now very much broken in this regard and has been for a few weeks. It would be good to solve the release first, and then re-address the solution for this ticket.

If I understand correctly that means the following PR's should be reverted:

@sleptor
Copy link
Author

sleptor commented Jul 9, 2024

@marcovtwout I'm afraid the only immediate solution is to downgrade to 2.0.49.4.

@samdark
Copy link
Member

samdark commented Jul 9, 2024

@terabytesoftw is on rolling back the changes.

@diego-mathis
Copy link
Contributor

This surely breaks even more stuff, totalcount is cached at runtime forever.

https://github.com/yiisoft/yii2/pull/20176/files#diff-097176e54cade8343dd0f3a488dc8269b8c5ea275a06abad5060183b4cf252f5R164

@terabytesoftw
Copy link
Member

terabytesoftw commented Jul 9, 2024

@terabytesoftw
Copy link
Member

I also think that testing needs to be improved.

@terabytesoftw
Copy link
Member

terabytesoftw commented Jul 10, 2024

@lav45 This is a new test, which is not even released and changes behavior that existed in framework since the beginning (and probably breaks BC in some cases) - I think it is fine to drop it if it creates any problems.

In general problems started with #20070 (preceded by #20007) and each attempt to fix them created even more problems and regressions. I suggest to revert all changes to state before #20070 or #20007 and start again from there. Messing with pagination and data provider cache looks like a dead end to me.

I think we should also revert this PR: #20002 @rob006 @samdark @sleptor

@samdark
Copy link
Member

samdark commented Jul 10, 2024

Alright.

@terabytesoftw
Copy link
Member

@rob006
Copy link
Contributor

rob006 commented Jul 10, 2024

@terabytesoftw It does not look like full revert, there are still changes comparing to 2.0.49:

protected function prepareTotalCount()
{
if (!$this->query instanceof QueryInterface) {
throw new InvalidConfigException('The "query" property must be an instance of a class that implements the QueryInterface e.g. yii\db\Query or its subclasses.');
}
$query = (clone $this->query)->limit(-1)->offset(-1)->orderBy([]);
$key = md5((string)$query);
if (!array_key_exists($key, $this->_totalCount)) {
$this->_totalCount[$key] = (int)$query->count('*', $this->db);
}
return $this->_totalCount[$key];
}

vs

protected function prepareTotalCount()
{
if (!$this->query instanceof QueryInterface) {
throw new InvalidConfigException('The "query" property must be an instance of a class that implements the QueryInterface e.g. yii\db\Query or its subclasses.');
}
$query = clone $this->query;
return (int) $query->limit(-1)->offset(-1)->orderBy([])->count('*', $this->db);
}

@terabytesoftw
Copy link
Member

@rob006 Fixed, thks.

@samdark samdark closed this as completed Jul 12, 2024
@schmunk42
Copy link
Contributor

Hi guys, sorry for being late to the party :)

JFTR, we were also affected by this bug.
Maybe next time we should move such a PR to the 2.2 branch.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

8 participants