From: | Anton Ratundalov <a(dot)ratundalov(at)postgrespro(dot)ru> |
---|---|
To: | pgsql-hackers(at)lists(dot)postgresql(dot)org |
Subject: | Remote partitionwise doesn't work for varchar join keys |
Date: | 2025-03-26 14:27:05 |
Message-ID: | b722a18c-78fc-4c99-ae9a-023960620e74@postgrespro.ru |
Views: | Raw Message | Whole Thread | Download mbox | Resend email |
Thread: | |
Lists: | pgsql-hackers |
Hi,
Faced a strange behaviour while attempting to join remote partitions
with varchar join keys.
Setup is as follows:
Servers : s1, s2, s3. s2 and s3 are linked to s1 via postgres_fdw as
foreign data sources.
Simple local tables on s2 and s3:
CREATE TABLE IF NOT exists tv1 (a int, b int, c varchar(40));
CREATE TABLE IF NOT exists tv2 (a int, b int, c varchar(40));
Partitioned tables on s1:
CREATE TABLE IF NOT exists tv1 (a int, b int, c varchar(40))
partition by hash(c);
CREATE TABLE IF NOT exists tv1_p1 partition of tv1
FOR values WITH (modulus 3, remainder 0);
CREATE FOREIGN TABLE IF NOT exists tv1_p2 partition of tv1
FOR values WITH (modulus 3, remainder 1) server s2 options (table_name
'tv1');
CREATE FOREIGN TABLE IF NOT exists tv1_p3 partition of tv1
FOR values WITH (modulus 3, remainder 2) server s3 options (table_name
'tv1');
CREATE TABLE IF NOT exists tv2 (a int, b int, c varchar(40))
partition by hash(c);
CREATE TABLE IF NOT exists tv2_p1 partition of tv2
FOR values WITH (modulus 3, remainder 0);
CREATE FOREIGN TABLE IF NOT exists tv2_p2 partition of tv2
FOR values WITH (modulus 3, remainder 1) server s2 options (table_name
'tv2');
CREATE FOREIGN TABLE IF NOT exists tv2_p3 partition of tv2
FOR values WITH (modulus 3, remainder 2) server s3 options (table_name
'tv2');
All the following queries are executed on s1.
Put some data to tv1, tv2:
INSERT INTO tv1 SELECT i, i, to_char(i/50, 'FM0000')
FROM generate_series(0, 599, 2) i;
INSERT INTO tv2 SELECT i, i, to_char(i/50, 'FM0000')
FROM generate_series(0, 599, 2) i;
Now, get a plan for partitionwise join on tv1.c = tv2.c
EXPLAIN(VERBOSE) SELECT tv1.a, tv2.a FROM tv1
JOIN tv2 ON tv1.c = tv2.c
WHERE tv2.c='0002';
=>
QUERY PLAN
----------------------------------------------------------------------
Nested Loop (cost=200.00..237.63 rows=9 width=8)
Output: tv1.a, tv2.a
-> Foreign Scan on public.tv1_p2 tv1 (cost=100.00..118.75 rows=3
width=102)
Output: tv1.a, tv1.c
Remote SQL: SELECT a, c FROM public.tv1 WHERE ((c = '0002'))
-> Materialize (cost=100.00..118.77 rows=3 width=102)
Output: tv2.a, tv2.c
-> Foreign Scan on public.tv2_p2 tv2 (cost=100.00..118.75
rows=3 width=102)
Output: tv2.a, tv2.c
Remote SQL: SELECT a, c FROM public.tv2 WHERE ((c = '0002'))
(10 rows)
If we append another remote partition to the query
EXPLAIN(VERBOSE) SELECT tv1.a, tv2.a
FROM tv1 JOIN tv2 ON tv1.c = tv2.c
WHERE tv2.c='0002' OR tv2.c='0000';
get a plan:
QUERY PLAN
----------------------------------------------------------------------
Append (cost=100.00..339.60 rows=40 width=8)
-> Foreign Scan (cost=100.00..169.70 rows=20 width=8)
Output: tv1_1.a, tv2_1.a
Relations: (public.tv1_p2 tv1_1) INNER JOIN (public.tv2_p2 tv2_1)
Remote SQL: SELECT r5.a, r7.a
FROM (public.tv1 r5 INNER JOIN public.tv2 r7 ON (((r5.c = r7.c))
AND (((r7.c = '0002') OR (r7.c = '0000')))))
-> Foreign Scan (cost=100.00..169.70 rows=20 width=8)
Output: tv1_2.a, tv2_2.a
Relations: (public.tv1_p3 tv1_2) INNER JOIN (public.tv2_p3 tv2_2)
Remote SQL: SELECT r6.a, r8.a
FROM (public.tv1 r6 INNER JOIN public.tv2 r8 ON (((r6.c = r8.c))
AND (((r8.c = '0002') OR (r8.c = '0000')))))
(9 rows)
If we substitute char() for varchar() join is pushed down to remote
server in both cases ( one and two remote partitions ).
To get varchar() keys behave the same way as char() the patch,
that you can find in attachment, is proposed.
Attachment | Content-Type | Size |
---|---|---|
remote_join_varchar.patch | text/x-patch | 929 bytes |
From | Date | Subject | |
---|---|---|---|
Next Message | Álvaro Herrera | 2025-03-26 14:30:49 | Re: vacuum_truncate configuration parameter and isset_offset |
Previous Message | Suraj Kharage | 2025-03-26 14:23:36 | Re: simplifying grammar for ALTER CONSTRAINT .. SET [NO] INHERIT |