Re: Slow plan choice with prepared query

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Dag, if you ain't right!  I can duplicate this on the ones I tested with: PG v11 and v14.  Gonna start diving into this myself...

Regards,
Michael Vitale


Mark Saward wrote on 2/23/2022 10:37 PM:
Hi,


I've experienced a situation where the planner seems to make a very poor choice with a prepared query after the first five executions.  Looking at the documentation, I think this happens because it switches from a custom plan to a generic one, and doesn't make a good choice for the generic one.

Postgres version: running in docker, reports to be 'Debian 14.1-1.pgdg110+1'

If I force it to use a custom plan via 'set local plan_cache_mode = force_custom_plan', then I don't notice any slowdown.  Without it, the 6th and onwards calls can take 1 second to 15 seconds each, as opposed to about 10ms.

Since I have a workaround, I don't necessarily need assistance, but posting this here in case it's of value as a test case. Here's a test case that reliably duplicates this issue for me:

----

create table test (
  test_id serial primary key,
  data text
);

insert into test (data) (select data from (select generate_series(1,10000) AS id, md5(random()::text) AS data) x);

prepare foo_test(text, text, int, text, bool) as SELECT * FROM (SELECT
  *,
  count(*) OVER () > $3 AS has_more,
  row_number() OVER ()
  FROM (
    WITH counted AS (
      SELECT count(*) AS total
      FROM   (select test_id::text, data
from test
where
  (cast($1 as text) is null or lower(data) like '%' || lower($1) || '%')
and
  (cast($2 as text) is null or lower(test_id::text) like '%' || lower($2) || '%')) base
    ), cursor_row AS (
      SELECT base.test_id
      FROM   (select test_id::text, data
from test
where
  (cast($1 as text) is null or lower(data) like '%' || lower($1) || '%')
and
  (cast($2 as text) is null or lower(test_id::text) like '%' || lower($2) || '%')) base
      WHERE  base.test_id = $4
    )
    SELECT counted.*, base.*
      FROM   (select test_id::text, data
from test
where
  (cast($1 as text) is null or lower(data) like '%' || lower($1) || '%')
and
  (cast($2 as text) is null or lower(test_id::text) like '%' || lower($2) || '%')) base
      LEFT JOIN   cursor_row ON true
      LEFT JOIN   counted ON true
      WHERE ((
            $4 IS NULL OR cast($5 as bool) IS NULL
          ) OR (
            (base.test_id)
              > (cursor_row.test_id)
          ))
      ORDER BY base.test_id ASC
      LIMIT $3 + 1
) xy LIMIT $3 ) z ORDER BY row_number ASC;

\timing

execute foo_test(null, null, 5, 500, true);
execute foo_test(null, null, 5, 500, true);
execute foo_test(null, null, 5, 500, true);
execute foo_test(null, null, 5, 500, true);
execute foo_test(null, null, 5, 500, true);

-- This one should be slower:
execute foo_test(null, null, 5, 500, true);









[Postgresql General]     [Postgresql PHP]     [PHP Users]     [PHP Home]     [PHP on Windows]     [Kernel Newbies]     [PHP Classes]     [PHP Books]     [PHP Databases]     [Yosemite]

  Powered by Linux