On one of my training sessions, I was presented with a simple query which looked like this:
SELECT company,
COUNT(*)
FROM invoices
WHERE can_access( company ) = 1
GROUP BY company;
COUNT(*)
FROM invoices
WHERE can_access( company ) = 1
GROUP BY company;
Execution plan with statistics was a follows:
Function can_access was containing a single SQL and some simple PL/SQL code. Purpose of this function was to filter records based on privileges of current user.
First, I told them, that I don’t like the usage of PL/SQL function at all in this kind of situation. It’s a CPU burner on SQL and PL/SQL context switches in first place, and second, it’s hiding some important information from Oracle optimizer (selectivity for example).
Response was that its legacy stuff and that they have to deal with it somehow … ouch. The problem was, that they knew that function could be ran on grouped result set (limiting calls by great deal), but they was unable to force Oracle to do so.
So I tried usual shenanigans with parentheses, no_merge, no_query_transformation and great deal of begging. But it was to no use. Execution plan looked always the same. In the end I used the following trick to do the job:
SELECT * FROM
(SELECT /*+ no_merge */
company,
COUNT(*)
FROM invoices
GROUP BY company)
WHERE (SELECT can_access( company ) FROM DUAL) = 1;
(SELECT /*+ no_merge */
company,
COUNT(*)
FROM invoices
GROUP BY company)
WHERE (SELECT can_access( company ) FROM DUAL) = 1;
Half processed buffers, nice time, great! After that I left home. Next day I’m telling my success story to a colleague of mine and he is like
“... Umm ... why don’t you use HAVING?”
So final solution should have looked like this:
SELECT company,
COUNT(*)
FROM invoices
GROUP BY company
HAVING can_access(company) = 1;
COUNT(*)
FROM invoices
GROUP BY company
HAVING can_access(company) = 1;
So remember folks … don’t try to solve stuff, when you are tired after a long day and think simple. You will spare yourself some facepalm.