123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325 |
- --
- -- JOIN
- -- Test JOIN clauses
- --
- CREATE TABLE J1_TBL (
- i integer,
- j integer,
- t text
- );
- CREATE TABLE J2_TBL (
- i integer,
- k integer
- );
- INSERT INTO J1_TBL VALUES (1, 4, 'one');
- INSERT INTO J1_TBL VALUES (2, 3, 'two');
- INSERT INTO J1_TBL VALUES (3, 2, 'three');
- INSERT INTO J1_TBL VALUES (4, 1, 'four');
- INSERT INTO J1_TBL VALUES (5, 0, 'five');
- INSERT INTO J1_TBL VALUES (6, 6, 'six');
- INSERT INTO J1_TBL VALUES (7, 7, 'seven');
- INSERT INTO J1_TBL VALUES (8, 8, 'eight');
- INSERT INTO J1_TBL VALUES (0, NULL, 'zero');
- INSERT INTO J1_TBL VALUES (NULL, NULL, 'null');
- INSERT INTO J1_TBL VALUES (NULL, 0, 'zero');
- INSERT INTO J2_TBL VALUES (1, -1);
- INSERT INTO J2_TBL VALUES (2, 2);
- INSERT INTO J2_TBL VALUES (3, -3);
- INSERT INTO J2_TBL VALUES (2, 4);
- INSERT INTO J2_TBL VALUES (5, -5);
- INSERT INTO J2_TBL VALUES (5, -5);
- INSERT INTO J2_TBL VALUES (0, NULL);
- INSERT INTO J2_TBL VALUES (NULL, NULL);
- INSERT INTO J2_TBL VALUES (NULL, 0);
- -- useful in some tests below
- create temp table onerow();
- insert into onerow default values;
- analyze onerow;
- --
- -- CORRELATION NAMES
- -- Make sure that table/column aliases are supported
- -- before diving into more complex join syntax.
- --
- SELECT *
- FROM J1_TBL AS tx;
- SELECT *
- FROM J1_TBL tx;
- SELECT *
- FROM J1_TBL AS t1 (a, b, c);
- SELECT *
- FROM J1_TBL t1 (a, b, c);
- SELECT *
- FROM J1_TBL t1 (a, b, c), J2_TBL t2 (d, e);
- SELECT t1.a, t2.e
- FROM J1_TBL t1 (a, b, c), J2_TBL t2 (d, e)
- WHERE t1.a = t2.d;
- --
- -- CROSS JOIN
- -- Qualifications are not allowed on cross joins,
- -- which degenerate into a standard unqualified inner join.
- --
- SELECT *
- FROM J1_TBL CROSS JOIN J2_TBL;
- -- ambiguous column
- SELECT i, k, t
- FROM J1_TBL CROSS JOIN J2_TBL;
- -- resolve previous ambiguity by specifying the table name
- SELECT t1.i, k, t
- FROM J1_TBL t1 CROSS JOIN J2_TBL t2;
- SELECT ii, tt, kk
- FROM (J1_TBL CROSS JOIN J2_TBL)
- AS tx (ii, jj, tt, ii2, kk);
- SELECT tx.ii, tx.jj, tx.kk
- FROM (J1_TBL t1 (a, b, c) CROSS JOIN J2_TBL t2 (d, e))
- AS tx (ii, jj, tt, ii2, kk);
- SELECT *
- FROM J1_TBL CROSS JOIN J2_TBL a CROSS JOIN J2_TBL b;
- --
- --
- -- Inner joins (equi-joins)
- --
- --
- --
- -- Inner joins (equi-joins) with USING clause
- -- The USING syntax changes the shape of the resulting table
- -- by including a column in the USING clause only once in the result.
- --
- -- Inner equi-join on specified column
- SELECT *
- FROM J1_TBL INNER JOIN J2_TBL USING (i);
- -- Same as above, slightly different syntax
- SELECT *
- FROM J1_TBL JOIN J2_TBL USING (i);
- SELECT *
- FROM J1_TBL t1 (a, b, c) JOIN J2_TBL t2 (a, d) USING (a)
- ORDER BY a, d;
- SELECT *
- FROM J1_TBL t1 (a, b, c) JOIN J2_TBL t2 (a, b) USING (b)
- ORDER BY b, t1.a;
- -- test join using aliases
- SELECT * FROM J1_TBL JOIN J2_TBL USING (i) WHERE J1_TBL.t = 'one'; -- ok
- SELECT * FROM J1_TBL JOIN J2_TBL USING (i) AS x WHERE J1_TBL.t = 'one'; -- ok
- SELECT * FROM (J1_TBL JOIN J2_TBL USING (i)) AS x WHERE J1_TBL.t = 'one'; -- error
- SELECT * FROM J1_TBL JOIN J2_TBL USING (i) AS x WHERE x.i = 1; -- ok
- SELECT * FROM J1_TBL JOIN J2_TBL USING (i) AS x WHERE x.t = 'one'; -- error
- SELECT * FROM (J1_TBL JOIN J2_TBL USING (i) AS x) AS xx WHERE x.i = 1; -- error (XXX could use better hint)
- SELECT * FROM J1_TBL a1 JOIN J2_TBL a2 USING (i) AS a1; -- error
- SELECT x.* FROM J1_TBL JOIN J2_TBL USING (i) AS x WHERE J1_TBL.t = 'one';
- SELECT ROW(x.*) FROM J1_TBL JOIN J2_TBL USING (i) AS x WHERE J1_TBL.t = 'one';
- SELECT row_to_json(x.*) FROM J1_TBL JOIN J2_TBL USING (i) AS x WHERE J1_TBL.t = 'one';
- --
- -- NATURAL JOIN
- -- Inner equi-join on all columns with the same name
- --
- SELECT *
- FROM J1_TBL NATURAL JOIN J2_TBL;
- SELECT *
- FROM J1_TBL t1 (a, b, c) NATURAL JOIN J2_TBL t2 (a, d);
- SELECT *
- FROM J1_TBL t1 (a, b, c) NATURAL JOIN J2_TBL t2 (d, a);
- -- mismatch number of columns
- -- currently, Postgres will fill in with underlying names
- SELECT *
- FROM J1_TBL t1 (a, b) NATURAL JOIN J2_TBL t2 (a);
- --
- -- Inner joins (equi-joins)
- --
- SELECT *
- FROM J1_TBL JOIN J2_TBL ON (J1_TBL.i = J2_TBL.i);
- SELECT *
- FROM J1_TBL JOIN J2_TBL ON (J1_TBL.i = J2_TBL.k);
- --
- -- Non-equi-joins
- --
- SELECT *
- FROM J1_TBL JOIN J2_TBL ON (J1_TBL.i <= J2_TBL.k);
- --
- -- Outer joins
- -- Note that OUTER is a noise word
- --
- SELECT *
- FROM J1_TBL LEFT OUTER JOIN J2_TBL USING (i)
- ORDER BY i, k, t;
- SELECT *
- FROM J1_TBL LEFT JOIN J2_TBL USING (i)
- ORDER BY i, k, t;
- SELECT *
- FROM J1_TBL RIGHT OUTER JOIN J2_TBL USING (i);
- SELECT *
- FROM J1_TBL RIGHT JOIN J2_TBL USING (i);
- SELECT *
- FROM J1_TBL FULL OUTER JOIN J2_TBL USING (i)
- ORDER BY i, k, t;
- SELECT *
- FROM J1_TBL FULL JOIN J2_TBL USING (i)
- ORDER BY i, k, t;
- SELECT *
- FROM J1_TBL LEFT JOIN J2_TBL USING (i) WHERE (k = 1);
- SELECT *
- FROM J1_TBL LEFT JOIN J2_TBL USING (i) WHERE (i = 1);
- --
- -- semijoin selectivity for <>
- --
- explain (costs off)
- select * from int4_tbl i4, tenk1 a
- where exists(select * from tenk1 b
- where a.twothousand = b.twothousand and a.fivethous <> b.fivethous)
- and i4.f1 = a.tenthous;
- --
- -- More complicated constructs
- --
- --
- -- Multiway full join
- --
- CREATE TABLE t1 (name TEXT, n INTEGER);
- CREATE TABLE t2 (name TEXT, n INTEGER);
- CREATE TABLE t3 (name TEXT, n INTEGER);
- INSERT INTO t1 VALUES ( 'bb', 11 );
- INSERT INTO t2 VALUES ( 'bb', 12 );
- INSERT INTO t2 VALUES ( 'cc', 22 );
- INSERT INTO t2 VALUES ( 'ee', 42 );
- INSERT INTO t3 VALUES ( 'bb', 13 );
- INSERT INTO t3 VALUES ( 'cc', 23 );
- INSERT INTO t3 VALUES ( 'dd', 33 );
- SELECT * FROM t1 FULL JOIN t2 USING (name) FULL JOIN t3 USING (name);
- --
- -- Test interactions of join syntax and subqueries
- --
- -- Basic cases (we expect planner to pull up the subquery here)
- SELECT * FROM
- (SELECT * FROM t2) as s2
- INNER JOIN
- (SELECT * FROM t3) s3
- USING (name);
- SELECT * FROM
- (SELECT * FROM t2) as s2
- LEFT JOIN
- (SELECT * FROM t3) s3
- USING (name);
- SELECT * FROM
- (SELECT * FROM t2) as s2
- FULL JOIN
- (SELECT * FROM t3) s3
- USING (name);
- -- Cases with non-nullable expressions in subquery results;
- -- make sure these go to null as expected
- SELECT * FROM
- (SELECT name, n as s2_n, 2 as s2_2 FROM t2) as s2
- NATURAL INNER JOIN
- (SELECT name, n as s3_n, 3 as s3_2 FROM t3) s3;
- SELECT * FROM
- (SELECT name, n as s2_n, 2 as s2_2 FROM t2) as s2
- NATURAL LEFT JOIN
- (SELECT name, n as s3_n, 3 as s3_2 FROM t3) s3;
- SELECT * FROM
- (SELECT name, n as s2_n, 2 as s2_2 FROM t2) as s2
- NATURAL FULL JOIN
- (SELECT name, n as s3_n, 3 as s3_2 FROM t3) s3;
- SELECT * FROM
- (SELECT name, n as s1_n, 1 as s1_1 FROM t1) as s1
- NATURAL INNER JOIN
- (SELECT name, n as s2_n, 2 as s2_2 FROM t2) as s2
- NATURAL INNER JOIN
- (SELECT name, n as s3_n, 3 as s3_2 FROM t3) s3;
- SELECT * FROM
- (SELECT name, n as s1_n, 1 as s1_1 FROM t1) as s1
- NATURAL FULL JOIN
- (SELECT name, n as s2_n, 2 as s2_2 FROM t2) as s2
- NATURAL FULL JOIN
- (SELECT name, n as s3_n, 3 as s3_2 FROM t3) s3;
- SELECT * FROM
- (SELECT name, n as s1_n FROM t1) as s1
- NATURAL FULL JOIN
- (SELECT * FROM
- (SELECT name, n as s2_n FROM t2) as s2
- NATURAL FULL JOIN
- (SELECT name, n as s3_n FROM t3) as s3
- ) ss2;
- SELECT * FROM
- (SELECT name, n as s1_n FROM t1) as s1
- NATURAL FULL JOIN
- (SELECT * FROM
- (SELECT name, n as s2_n, 2 as s2_2 FROM t2) as s2
- NATURAL FULL JOIN
- (SELECT name, n as s3_n FROM t3) as s3
- ) ss2;
- -- Constants as join keys can also be problematic
- SELECT * FROM
- (SELECT name, n as s1_n FROM t1) as s1
- FULL JOIN
- (SELECT name, 2 as s2_n FROM t2) as s2
- ON (s1_n = s2_n);
- -- Test for propagation of nullability constraints into sub-joins
- create temp table x (x1 int, x2 int);
- insert into x values (1,11);
- insert into x values (2,22);
- insert into x values (3,null);
- insert into x values (4,44);
- insert into x values (5,null);
- create temp table y (y1 int, y2 int);
- insert into y values (1,111);
- insert into y values (2,222);
- insert into y values (3,333);
- insert into y values (4,null);
- select * from x;
- select * from y;
- select * from x left join y on (x1 = y1 and x2 is not null);
- select * from x left join y on (x1 = y1 and y2 is not null);
- select * from (x left join y on (x1 = y1)) left join x xx(xx1,xx2)
- on (x1 = xx1);
- select * from (x left join y on (x1 = y1)) left join x xx(xx1,xx2)
- on (x1 = xx1 and x2 is not null);
- select * from (x left join y on (x1 = y1)) left join x xx(xx1,xx2)
- on (x1 = xx1 and y2 is not null);
- select * from (x left join y on (x1 = y1)) left join x xx(xx1,xx2)
- on (x1 = xx1 and xx2 is not null);
- -- these should NOT give the same answers as above
- select * from (x left join y on (x1 = y1)) left join x xx(xx1,xx2)
- on (x1 = xx1) where (x2 is not null);
- select * from (x left join y on (x1 = y1)) left join x xx(xx1,xx2)
- on (x1 = xx1) where (y2 is not null);
- select * from (x left join y on (x1 = y1)) left join x xx(xx1,xx2)
- on (x1 = xx1) where (xx2 is not null);
- --
- -- regression test: check for bug with propagation of implied equality
- -- to outside an IN
- --
- select count(*) from tenk1 a where unique1 in
- (select unique1 from tenk1 b join tenk1 c using (unique1)
- where b.unique2 = 42);
- --
- -- regression test: check for failure to generate a plan with multiple
- -- degenerate IN clauses
- --
- select count(*) from tenk1 x where
- x.unique1 in (select a.f1 from int4_tbl a,float8_tbl b where a.f1=b.f1) and
- x.unique1 = 0 and
- x.unique1 in (select aa.f1 from int4_tbl aa,float8_tbl bb where aa.f1=bb.f1);
- -- try that with GEQO too
- begin;
- set geqo = on;
- set geqo_threshold = 2;
- select count(*) from tenk1 x where
- x.unique1 in (select a.f1 from int4_tbl a,float8_tbl b where a.f1=b.f1) and
- x.unique1 = 0 and
- x.unique1 in (select aa.f1 from int4_tbl aa,float8_tbl bb where aa.f1=bb.f1);
- rollback;
- --
- -- regression test: be sure we cope with proven-dummy append rels
- --
- explain (costs off)
- select aa, bb, unique1, unique1
- from tenk1 right join b on aa = unique1
- where bb < bb and bb is null;
- select aa, bb, unique1, unique1
- from tenk1 right join b on aa = unique1
- where bb < bb and bb is null;
- --
- -- regression test: check handling of empty-FROM subquery underneath outer join
- --
- explain (costs off)
- select * from int8_tbl i1 left join (int8_tbl i2 join
- (select 123 as x) ss on i2.q1 = x) on i1.q2 = i2.q2
- order by 1, 2;
- select * from int8_tbl i1 left join (int8_tbl i2 join
- (select 123 as x) ss on i2.q1 = x) on i1.q2 = i2.q2
- order by 1, 2;
- --
- -- regression test: check a case where join_clause_is_movable_into() gives
- -- an imprecise result, causing an assertion failure
- --
- select count(*)
- from
- (select t3.tenthous as x1, coalesce(t1.stringu1, t2.stringu1) as x2
- from tenk1 t1
- left join tenk1 t2 on t1.unique1 = t2.unique1
- join tenk1 t3 on t1.unique2 = t3.unique2) ss,
- tenk1 t4,
- tenk1 t5
- where t4.thousand = t5.unique1 and ss.x1 = t4.tenthous and ss.x2 = t5.stringu1;
- --
- -- regression test: check a case where we formerly missed including an EC
- -- enforcement clause because it was expected to be handled at scan level
- --
- explain (costs off)
- select a.f1, b.f1, t.thousand, t.tenthous from
- tenk1 t,
- (select sum(f1)+1 as f1 from int4_tbl i4a) a,
- (select sum(f1) as f1 from int4_tbl i4b) b
- where b.f1 = t.thousand and a.f1 = b.f1 and (a.f1+b.f1+999) = t.tenthous;
- select a.f1, b.f1, t.thousand, t.tenthous from
- tenk1 t,
- (select sum(f1)+1 as f1 from int4_tbl i4a) a,
- (select sum(f1) as f1 from int4_tbl i4b) b
- where b.f1 = t.thousand and a.f1 = b.f1 and (a.f1+b.f1+999) = t.tenthous;
- --
- -- check a case where we formerly got confused by conflicting sort orders
- -- in redundant merge join path keys
- --
- explain (costs off)
- select * from
- j1_tbl full join
- (select * from j2_tbl order by j2_tbl.i desc, j2_tbl.k asc) j2_tbl
- on j1_tbl.i = j2_tbl.i and j1_tbl.i = j2_tbl.k;
- select * from
- j1_tbl full join
- (select * from j2_tbl order by j2_tbl.i desc, j2_tbl.k asc) j2_tbl
- on j1_tbl.i = j2_tbl.i and j1_tbl.i = j2_tbl.k;
- --
- -- a different check for handling of redundant sort keys in merge joins
- --
- explain (costs off)
- select count(*) from
- (select * from tenk1 x order by x.thousand, x.twothousand, x.fivethous) x
- left join
- (select * from tenk1 y order by y.unique2) y
- on x.thousand = y.unique2 and x.twothousand = y.hundred and x.fivethous = y.unique2;
- select count(*) from
- (select * from tenk1 x order by x.thousand, x.twothousand, x.fivethous) x
- left join
- (select * from tenk1 y order by y.unique2) y
- on x.thousand = y.unique2 and x.twothousand = y.hundred and x.fivethous = y.unique2;
- --
- -- Clean up
- --
- DROP TABLE t1;
- DROP TABLE t2;
- DROP TABLE t3;
- DROP TABLE J1_TBL;
- DROP TABLE J2_TBL;
- -- Both DELETE and UPDATE allow the specification of additional tables
- -- to "join" against to determine which rows should be modified.
- CREATE TEMP TABLE t1 (a int, b int);
- CREATE TEMP TABLE t2 (a int, b int);
- CREATE TEMP TABLE t3 (x int, y int);
- INSERT INTO t1 VALUES (5, 10);
- INSERT INTO t1 VALUES (15, 20);
- INSERT INTO t1 VALUES (100, 100);
- INSERT INTO t1 VALUES (200, 1000);
- INSERT INTO t2 VALUES (200, 2000);
- INSERT INTO t3 VALUES (5, 20);
- INSERT INTO t3 VALUES (6, 7);
- INSERT INTO t3 VALUES (7, 8);
- INSERT INTO t3 VALUES (500, 100);
- DELETE FROM t3 USING t1 table1 WHERE t3.x = table1.a;
- SELECT * FROM t3;
- DELETE FROM t3 USING t1 JOIN t2 USING (a) WHERE t3.x > t1.a;
- SELECT * FROM t3;
- DELETE FROM t3 USING t3 t3_other WHERE t3.x = t3_other.x AND t3.y = t3_other.y;
- SELECT * FROM t3;
- -- Test join against inheritance tree
- create temp table t2a () inherits (t2);
- insert into t2a values (200, 2001);
- select * from t1 left join t2 on (t1.a = t2.a);
- -- Test matching of column name with wrong alias
- select t1.x from t1 join t3 on (t1.a = t3.x);
- -- Test matching of locking clause with wrong alias
- select t1.*, t2.*, unnamed_join.* from
- t1 join t2 on (t1.a = t2.a), t3 as unnamed_join
- for update of unnamed_join;
- select foo.*, unnamed_join.* from
- t1 join t2 using (a) as foo, t3 as unnamed_join
- for update of unnamed_join;
- select foo.*, unnamed_join.* from
- t1 join t2 using (a) as foo, t3 as unnamed_join
- for update of foo;
- select bar.*, unnamed_join.* from
- (t1 join t2 using (a) as foo) as bar, t3 as unnamed_join
- for update of foo;
- select bar.*, unnamed_join.* from
- (t1 join t2 using (a) as foo) as bar, t3 as unnamed_join
- for update of bar;
- --
- -- regression test for 8.1 merge right join bug
- --
- CREATE TEMP TABLE tt1 ( tt1_id int4, joincol int4 );
- INSERT INTO tt1 VALUES (1, 11);
- INSERT INTO tt1 VALUES (2, NULL);
- CREATE TEMP TABLE tt2 ( tt2_id int4, joincol int4 );
- INSERT INTO tt2 VALUES (21, 11);
- INSERT INTO tt2 VALUES (22, 11);
- set enable_hashjoin to off;
- set enable_nestloop to off;
- -- these should give the same results
- select tt1.*, tt2.* from tt1 left join tt2 on tt1.joincol = tt2.joincol;
- select tt1.*, tt2.* from tt2 right join tt1 on tt1.joincol = tt2.joincol;
- reset enable_hashjoin;
- reset enable_nestloop;
- --
- -- regression test for bug #13908 (hash join with skew tuples & nbatch increase)
- --
- set work_mem to '64kB';
- set enable_mergejoin to off;
- set enable_memoize to off;
- explain (costs off)
- select count(*) from tenk1 a, tenk1 b
- where a.hundred = b.thousand and (b.fivethous % 10) < 10;
- select count(*) from tenk1 a, tenk1 b
- where a.hundred = b.thousand and (b.fivethous % 10) < 10;
- reset work_mem;
- reset enable_mergejoin;
- reset enable_memoize;
- --
- -- regression test for 8.2 bug with improper re-ordering of left joins
- --
- create temp table tt3(f1 int, f2 text);
- insert into tt3 select x, repeat('xyzzy', 100) from generate_series(1,10000) x;
- create index tt3i on tt3(f1);
- analyze tt3;
- create temp table tt4(f1 int);
- insert into tt4 values (0),(1),(9999);
- analyze tt4;
- SELECT a.f1
- FROM tt4 a
- LEFT JOIN (
- SELECT b.f1
- FROM tt3 b LEFT JOIN tt3 c ON (b.f1 = c.f1)
- WHERE c.f1 IS NULL
- ) AS d ON (a.f1 = d.f1)
- WHERE d.f1 IS NULL;
- --
- -- regression test for proper handling of outer joins within antijoins
- --
- create temp table tt4x(c1 int, c2 int, c3 int);
- explain (costs off)
- select * from tt4x t1
- where not exists (
- select 1 from tt4x t2
- left join tt4x t3 on t2.c3 = t3.c1
- left join ( select t5.c1 as c1
- from tt4x t4 left join tt4x t5 on t4.c2 = t5.c1
- ) a1 on t3.c2 = a1.c1
- where t1.c1 = t2.c2
- );
- --
- -- regression test for problems of the sort depicted in bug #3494
- --
- create temp table tt5(f1 int, f2 int);
- create temp table tt6(f1 int, f2 int);
- insert into tt5 values(1, 10);
- insert into tt5 values(1, 11);
- insert into tt6 values(1, 9);
- insert into tt6 values(1, 2);
- insert into tt6 values(2, 9);
- select * from tt5,tt6 where tt5.f1 = tt6.f1 and tt5.f1 = tt5.f2 - tt6.f2;
- --
- -- regression test for problems of the sort depicted in bug #3588
- --
- create temp table xx (pkxx int);
- create temp table yy (pkyy int, pkxx int);
- insert into xx values (1);
- insert into xx values (2);
- insert into xx values (3);
- insert into yy values (101, 1);
- insert into yy values (201, 2);
- insert into yy values (301, NULL);
- select yy.pkyy as yy_pkyy, yy.pkxx as yy_pkxx, yya.pkyy as yya_pkyy,
- xxa.pkxx as xxa_pkxx, xxb.pkxx as xxb_pkxx
- from yy
- left join (SELECT * FROM yy where pkyy = 101) as yya ON yy.pkyy = yya.pkyy
- left join xx xxa on yya.pkxx = xxa.pkxx
- left join xx xxb on coalesce (xxa.pkxx, 1) = xxb.pkxx;
- --
- -- regression test for improper pushing of constants across outer-join clauses
- -- (as seen in early 8.2.x releases)
- --
- create temp table zt1 (f1 int primary key);
- create temp table zt2 (f2 int primary key);
- create temp table zt3 (f3 int primary key);
- insert into zt1 values(53);
- insert into zt2 values(53);
- select * from
- zt2 left join zt3 on (f2 = f3)
- left join zt1 on (f3 = f1)
- where f2 = 53;
- create temp view zv1 as select *,'dummy'::text AS junk from zt1;
- select * from
- zt2 left join zt3 on (f2 = f3)
- left join zv1 on (f3 = f1)
- where f2 = 53;
- --
- -- regression test for improper extraction of OR indexqual conditions
- -- (as seen in early 8.3.x releases)
- --
- select a.unique2, a.ten, b.tenthous, b.unique2, b.hundred
- from tenk1 a left join tenk1 b on a.unique2 = b.tenthous
- where a.unique1 = 42 and
- ((b.unique2 is null and a.ten = 2) or b.hundred = 3);
- --
- -- test proper positioning of one-time quals in EXISTS (8.4devel bug)
- --
- prepare foo(bool) as
- select count(*) from tenk1 a left join tenk1 b
- on (a.unique2 = b.unique1 and exists
- (select 1 from tenk1 c where c.thousand = b.unique2 and $1));
- execute foo(true);
- execute foo(false);
- --
- -- test for sane behavior with noncanonical merge clauses, per bug #4926
- --
- begin;
- set enable_mergejoin = 1;
- set enable_hashjoin = 0;
- set enable_nestloop = 0;
- create temp table a (i integer);
- create temp table b (x integer, y integer);
- select * from a left join b on i = x and i = y and x = i;
- rollback;
- --
- -- test handling of merge clauses using record_ops
- --
- begin;
- create type mycomptype as (id int, v bigint);
- create temp table tidv (idv mycomptype);
- create index on tidv (idv);
- explain (costs off)
- select a.idv, b.idv from tidv a, tidv b where a.idv = b.idv;
- set enable_mergejoin = 0;
- set enable_hashjoin = 0;
- explain (costs off)
- select a.idv, b.idv from tidv a, tidv b where a.idv = b.idv;
- rollback;
- --
- -- test NULL behavior of whole-row Vars, per bug #5025
- --
- select t1.q2, count(t2.*)
- from int8_tbl t1 left join int8_tbl t2 on (t1.q2 = t2.q1)
- group by t1.q2 order by 1;
- select t1.q2, count(t2.*)
- from int8_tbl t1 left join (select * from int8_tbl) t2 on (t1.q2 = t2.q1)
- group by t1.q2 order by 1;
- select t1.q2, count(t2.*)
- from int8_tbl t1 left join (select * from int8_tbl offset 0) t2 on (t1.q2 = t2.q1)
- group by t1.q2 order by 1;
- select t1.q2, count(t2.*)
- from int8_tbl t1 left join
- (select q1, case when q2=1 then 1 else q2 end as q2 from int8_tbl) t2
- on (t1.q2 = t2.q1)
- group by t1.q2 order by 1;
- --
- -- test incorrect failure to NULL pulled-up subexpressions
- --
- begin;
- create temp table a (
- code char not null,
- constraint a_pk primary key (code)
- );
- create temp table b (
- a char not null,
- num integer not null,
- constraint b_pk primary key (a, num)
- );
- create temp table c (
- name char not null,
- a char,
- constraint c_pk primary key (name)
- );
- insert into a (code) values ('p');
- insert into a (code) values ('q');
- insert into b (a, num) values ('p', 1);
- insert into b (a, num) values ('p', 2);
- insert into c (name, a) values ('A', 'p');
- insert into c (name, a) values ('B', 'q');
- insert into c (name, a) values ('C', null);
- select c.name, ss.code, ss.b_cnt, ss.const
- from c left join
- (select a.code, coalesce(b_grp.cnt, 0) as b_cnt, -1 as const
- from a left join
- (select count(1) as cnt, b.a from b group by b.a) as b_grp
- on a.code = b_grp.a
- ) as ss
- on (c.a = ss.code)
- order by c.name;
- rollback;
- --
- -- test incorrect handling of placeholders that only appear in targetlists,
- -- per bug #6154
- --
- SELECT * FROM
- ( SELECT 1 as key1 ) sub1
- LEFT JOIN
- ( SELECT sub3.key3, sub4.value2, COALESCE(sub4.value2, 66) as value3 FROM
- ( SELECT 1 as key3 ) sub3
- LEFT JOIN
- ( SELECT sub5.key5, COALESCE(sub6.value1, 1) as value2 FROM
- ( SELECT 1 as key5 ) sub5
- LEFT JOIN
- ( SELECT 2 as key6, 42 as value1 ) sub6
- ON sub5.key5 = sub6.key6
- ) sub4
- ON sub4.key5 = sub3.key3
- ) sub2
- ON sub1.key1 = sub2.key3;
- -- test the path using join aliases, too
- SELECT * FROM
- ( SELECT 1 as key1 ) sub1
- LEFT JOIN
- ( SELECT sub3.key3, value2, COALESCE(value2, 66) as value3 FROM
- ( SELECT 1 as key3 ) sub3
- LEFT JOIN
- ( SELECT sub5.key5, COALESCE(sub6.value1, 1) as value2 FROM
- ( SELECT 1 as key5 ) sub5
- LEFT JOIN
- ( SELECT 2 as key6, 42 as value1 ) sub6
- ON sub5.key5 = sub6.key6
- ) sub4
- ON sub4.key5 = sub3.key3
- ) sub2
- ON sub1.key1 = sub2.key3;
- --
- -- test case where a PlaceHolderVar is used as a nestloop parameter
- --
- EXPLAIN (COSTS OFF)
- SELECT qq, unique1
- FROM
- ( SELECT COALESCE(q1, 0) AS qq FROM int8_tbl a ) AS ss1
- FULL OUTER JOIN
- ( SELECT COALESCE(q2, -1) AS qq FROM int8_tbl b ) AS ss2
- USING (qq)
- INNER JOIN tenk1 c ON qq = unique2;
- SELECT qq, unique1
- FROM
- ( SELECT COALESCE(q1, 0) AS qq FROM int8_tbl a ) AS ss1
- FULL OUTER JOIN
- ( SELECT COALESCE(q2, -1) AS qq FROM int8_tbl b ) AS ss2
- USING (qq)
- INNER JOIN tenk1 c ON qq = unique2;
- --
- -- nested nestloops can require nested PlaceHolderVars
- --
- create temp table nt1 (
- id int primary key,
- a1 boolean,
- a2 boolean
- );
- create temp table nt2 (
- id int primary key,
- nt1_id int,
- b1 boolean,
- b2 boolean,
- foreign key (nt1_id) references nt1(id)
- );
- create temp table nt3 (
- id int primary key,
- nt2_id int,
- c1 boolean,
- foreign key (nt2_id) references nt2(id)
- );
- insert into nt1 values (1,true,true);
- insert into nt1 values (2,true,false);
- insert into nt1 values (3,false,false);
- insert into nt2 values (1,1,true,true);
- insert into nt2 values (2,2,true,false);
- insert into nt2 values (3,3,false,false);
- insert into nt3 values (1,1,true);
- insert into nt3 values (2,2,false);
- insert into nt3 values (3,3,true);
- explain (costs off)
- select nt3.id
- from nt3 as nt3
- left join
- (select nt2.*, (nt2.b1 and ss1.a3) AS b3
- from nt2 as nt2
- left join
- (select nt1.*, (nt1.id is not null) as a3 from nt1) as ss1
- on ss1.id = nt2.nt1_id
- ) as ss2
- on ss2.id = nt3.nt2_id
- where nt3.id = 1 and ss2.b3;
- select nt3.id
- from nt3 as nt3
- left join
- (select nt2.*, (nt2.b1 and ss1.a3) AS b3
- from nt2 as nt2
- left join
- (select nt1.*, (nt1.id is not null) as a3 from nt1) as ss1
- on ss1.id = nt2.nt1_id
- ) as ss2
- on ss2.id = nt3.nt2_id
- where nt3.id = 1 and ss2.b3;
- --
- -- test case where a PlaceHolderVar is propagated into a subquery
- --
- explain (costs off)
- select * from
- int8_tbl t1 left join
- (select q1 as x, 42 as y from int8_tbl t2) ss
- on t1.q2 = ss.x
- where
- 1 = (select 1 from int8_tbl t3 where ss.y is not null limit 1)
- order by 1,2;
- select * from
- int8_tbl t1 left join
- (select q1 as x, 42 as y from int8_tbl t2) ss
- on t1.q2 = ss.x
- where
- 1 = (select 1 from int8_tbl t3 where ss.y is not null limit 1)
- order by 1,2;
- --
- -- variant where a PlaceHolderVar is needed at a join, but not above the join
- --
- explain (costs off)
- select * from
- int4_tbl as i41,
- lateral
- (select 1 as x from
- (select i41.f1 as lat,
- i42.f1 as loc from
- int8_tbl as i81, int4_tbl as i42) as ss1
- right join int4_tbl as i43 on (i43.f1 > 1)
- where ss1.loc = ss1.lat) as ss2
- where i41.f1 > 0;
- select * from
- int4_tbl as i41,
- lateral
- (select 1 as x from
- (select i41.f1 as lat,
- i42.f1 as loc from
- int8_tbl as i81, int4_tbl as i42) as ss1
- right join int4_tbl as i43 on (i43.f1 > 1)
- where ss1.loc = ss1.lat) as ss2
- where i41.f1 > 0;
- --
- -- test the corner cases FULL JOIN ON TRUE and FULL JOIN ON FALSE
- --
- select * from int4_tbl a full join int4_tbl b on true;
- select * from int4_tbl a full join int4_tbl b on false;
- --
- -- test for ability to use a cartesian join when necessary
- --
- create temp table q1 as select 1 as q1;
- create temp table q2 as select 0 as q2;
- analyze q1;
- analyze q2;
- explain (costs off)
- select * from
- tenk1 join int4_tbl on f1 = twothousand,
- q1, q2
- where q1 = thousand or q2 = thousand;
- explain (costs off)
- select * from
- tenk1 join int4_tbl on f1 = twothousand,
- q1, q2
- where thousand = (q1 + q2);
- --
- -- test ability to generate a suitable plan for a star-schema query
- --
- explain (costs off)
- select * from
- tenk1, int8_tbl a, int8_tbl b
- where thousand = a.q1 and tenthous = b.q1 and a.q2 = 1 and b.q2 = 2;
- --
- -- test a corner case in which we shouldn't apply the star-schema optimization
- --
- explain (costs off)
- select t1.unique2, t1.stringu1, t2.unique1, t2.stringu2 from
- tenk1 t1
- inner join int4_tbl i1
- left join (select v1.x2, v2.y1, 11 AS d1
- from (select 1,0 from onerow) v1(x1,x2)
- left join (select 3,1 from onerow) v2(y1,y2)
- on v1.x1 = v2.y2) subq1
- on (i1.f1 = subq1.x2)
- on (t1.unique2 = subq1.d1)
- left join tenk1 t2
- on (subq1.y1 = t2.unique1)
- where t1.unique2 < 42 and t1.stringu1 > t2.stringu2;
- select t1.unique2, t1.stringu1, t2.unique1, t2.stringu2 from
- tenk1 t1
- inner join int4_tbl i1
- left join (select v1.x2, v2.y1, 11 AS d1
- from (select 1,0 from onerow) v1(x1,x2)
- left join (select 3,1 from onerow) v2(y1,y2)
- on v1.x1 = v2.y2) subq1
- on (i1.f1 = subq1.x2)
- on (t1.unique2 = subq1.d1)
- left join tenk1 t2
- on (subq1.y1 = t2.unique1)
- where t1.unique2 < 42 and t1.stringu1 > t2.stringu2;
- -- variant that isn't quite a star-schema case
- select ss1.d1 from
- tenk1 as t1
- inner join tenk1 as t2
- on t1.tenthous = t2.ten
- inner join
- int8_tbl as i8
- left join int4_tbl as i4
- inner join (select 64::information_schema.cardinal_number as d1
- from tenk1 t3,
- lateral (select abs(t3.unique1) + random()) ss0(x)
- where t3.fivethous < 0) as ss1
- on i4.f1 = ss1.d1
- on i8.q1 = i4.f1
- on t1.tenthous = ss1.d1
- where t1.unique1 < i4.f1;
- -- this variant is foldable by the remove-useless-RESULT-RTEs code
- explain (costs off)
- select t1.unique2, t1.stringu1, t2.unique1, t2.stringu2 from
- tenk1 t1
- inner join int4_tbl i1
- left join (select v1.x2, v2.y1, 11 AS d1
- from (values(1,0)) v1(x1,x2)
- left join (values(3,1)) v2(y1,y2)
- on v1.x1 = v2.y2) subq1
- on (i1.f1 = subq1.x2)
- on (t1.unique2 = subq1.d1)
- left join tenk1 t2
- on (subq1.y1 = t2.unique1)
- where t1.unique2 < 42 and t1.stringu1 > t2.stringu2;
- select t1.unique2, t1.stringu1, t2.unique1, t2.stringu2 from
- tenk1 t1
- inner join int4_tbl i1
- left join (select v1.x2, v2.y1, 11 AS d1
- from (values(1,0)) v1(x1,x2)
- left join (values(3,1)) v2(y1,y2)
- on v1.x1 = v2.y2) subq1
- on (i1.f1 = subq1.x2)
- on (t1.unique2 = subq1.d1)
- left join tenk1 t2
- on (subq1.y1 = t2.unique1)
- where t1.unique2 < 42 and t1.stringu1 > t2.stringu2;
- -- Here's a variant that we can't fold too aggressively, though,
- -- or we end up with noplace to evaluate the lateral PHV
- explain (verbose, costs off)
- select * from
- (select 1 as x) ss1 left join (select 2 as y) ss2 on (true),
- lateral (select ss2.y as z limit 1) ss3;
- select * from
- (select 1 as x) ss1 left join (select 2 as y) ss2 on (true),
- lateral (select ss2.y as z limit 1) ss3;
- -- Test proper handling of appendrel PHVs during useless-RTE removal
- explain (costs off)
- select * from
- (select 0 as z) as t1
- left join
- (select true as a) as t2
- on true,
- lateral (select true as b
- union all
- select a as b) as t3
- where b;
- select * from
- (select 0 as z) as t1
- left join
- (select true as a) as t2
- on true,
- lateral (select true as b
- union all
- select a as b) as t3
- where b;
- --
- -- test inlining of immutable functions
- --
- create function f_immutable_int4(i integer) returns integer as
- $$ begin return i; end; $$ language plpgsql immutable;
- -- check optimization of function scan with join
- explain (costs off)
- select unique1 from tenk1, (select * from f_immutable_int4(1) x) x
- where x = unique1;
- explain (verbose, costs off)
- select unique1, x.*
- from tenk1, (select *, random() from f_immutable_int4(1) x) x
- where x = unique1;
- explain (costs off)
- select unique1 from tenk1, f_immutable_int4(1) x where x = unique1;
- explain (costs off)
- select unique1 from tenk1, lateral f_immutable_int4(1) x where x = unique1;
- explain (costs off)
- select unique1 from tenk1, lateral f_immutable_int4(1) x where x in (select 17);
- explain (costs off)
- select unique1, x from tenk1 join f_immutable_int4(1) x on unique1 = x;
- explain (costs off)
- select unique1, x from tenk1 left join f_immutable_int4(1) x on unique1 = x;
- explain (costs off)
- select unique1, x from tenk1 right join f_immutable_int4(1) x on unique1 = x;
- explain (costs off)
- select unique1, x from tenk1 full join f_immutable_int4(1) x on unique1 = x;
- -- check that pullup of a const function allows further const-folding
- explain (costs off)
- select unique1 from tenk1, f_immutable_int4(1) x where x = 42;
- -- test inlining of immutable functions with PlaceHolderVars
- explain (costs off)
- select nt3.id
- from nt3 as nt3
- left join
- (select nt2.*, (nt2.b1 or i4 = 42) AS b3
- from nt2 as nt2
- left join
- f_immutable_int4(0) i4
- on i4 = nt2.nt1_id
- ) as ss2
- on ss2.id = nt3.nt2_id
- where nt3.id = 1 and ss2.b3;
- drop function f_immutable_int4(int);
- -- test inlining when function returns composite
- create function mki8(bigint, bigint) returns int8_tbl as
- $$select row($1,$2)::int8_tbl$$ language sql;
- create function mki4(int) returns int4_tbl as
- $$select row($1)::int4_tbl$$ language sql;
- explain (verbose, costs off)
- select * from mki8(1,2);
- select * from mki8(1,2);
- explain (verbose, costs off)
- select * from mki4(42);
- select * from mki4(42);
- drop function mki8(bigint, bigint);
- drop function mki4(int);
- --
- -- test extraction of restriction OR clauses from join OR clause
- -- (we used to only do this for indexable clauses)
- --
- explain (costs off)
- select * from tenk1 a join tenk1 b on
- (a.unique1 = 1 and b.unique1 = 2) or (a.unique2 = 3 and b.hundred = 4);
- explain (costs off)
- select * from tenk1 a join tenk1 b on
- (a.unique1 = 1 and b.unique1 = 2) or (a.unique2 = 3 and b.ten = 4);
- explain (costs off)
- select * from tenk1 a join tenk1 b on
- (a.unique1 = 1 and b.unique1 = 2) or
- ((a.unique2 = 3 or a.unique2 = 7) and b.hundred = 4);
- --
- -- test placement of movable quals in a parameterized join tree
- --
- explain (costs off)
- select * from tenk1 t1 left join
- (tenk1 t2 join tenk1 t3 on t2.thousand = t3.unique2)
- on t1.hundred = t2.hundred and t1.ten = t3.ten
- where t1.unique1 = 1;
- explain (costs off)
- select * from tenk1 t1 left join
- (tenk1 t2 join tenk1 t3 on t2.thousand = t3.unique2)
- on t1.hundred = t2.hundred and t1.ten + t2.ten = t3.ten
- where t1.unique1 = 1;
- explain (costs off)
- select count(*) from
- tenk1 a join tenk1 b on a.unique1 = b.unique2
- left join tenk1 c on a.unique2 = b.unique1 and c.thousand = a.thousand
- join int4_tbl on b.thousand = f1;
- select count(*) from
- tenk1 a join tenk1 b on a.unique1 = b.unique2
- left join tenk1 c on a.unique2 = b.unique1 and c.thousand = a.thousand
- join int4_tbl on b.thousand = f1;
- explain (costs off)
- select b.unique1 from
- tenk1 a join tenk1 b on a.unique1 = b.unique2
- left join tenk1 c on b.unique1 = 42 and c.thousand = a.thousand
- join int4_tbl i1 on b.thousand = f1
- right join int4_tbl i2 on i2.f1 = b.tenthous
- order by 1;
- select b.unique1 from
- tenk1 a join tenk1 b on a.unique1 = b.unique2
- left join tenk1 c on b.unique1 = 42 and c.thousand = a.thousand
- join int4_tbl i1 on b.thousand = f1
- right join int4_tbl i2 on i2.f1 = b.tenthous
- order by 1;
- explain (costs off)
- select * from
- (
- select unique1, q1, coalesce(unique1, -1) + q1 as fault
- from int8_tbl left join tenk1 on (q2 = unique2)
- ) ss
- where fault = 122
- order by fault;
- select * from
- (
- select unique1, q1, coalesce(unique1, -1) + q1 as fault
- from int8_tbl left join tenk1 on (q2 = unique2)
- ) ss
- where fault = 122
- order by fault;
- explain (costs off)
- select * from
- (values (1, array[10,20]), (2, array[20,30])) as v1(v1x,v1ys)
- left join (values (1, 10), (2, 20)) as v2(v2x,v2y) on v2x = v1x
- left join unnest(v1ys) as u1(u1y) on u1y = v2y;
- select * from
- (values (1, array[10,20]), (2, array[20,30])) as v1(v1x,v1ys)
- left join (values (1, 10), (2, 20)) as v2(v2x,v2y) on v2x = v1x
- left join unnest(v1ys) as u1(u1y) on u1y = v2y;
- --
- -- test handling of potential equivalence clauses above outer joins
- --
- explain (costs off)
- select q1, unique2, thousand, hundred
- from int8_tbl a left join tenk1 b on q1 = unique2
- where coalesce(thousand,123) = q1 and q1 = coalesce(hundred,123);
- select q1, unique2, thousand, hundred
- from int8_tbl a left join tenk1 b on q1 = unique2
- where coalesce(thousand,123) = q1 and q1 = coalesce(hundred,123);
- explain (costs off)
- select f1, unique2, case when unique2 is null then f1 else 0 end
- from int4_tbl a left join tenk1 b on f1 = unique2
- where (case when unique2 is null then f1 else 0 end) = 0;
- select f1, unique2, case when unique2 is null then f1 else 0 end
- from int4_tbl a left join tenk1 b on f1 = unique2
- where (case when unique2 is null then f1 else 0 end) = 0;
- --
- -- another case with equivalence clauses above outer joins (bug #8591)
- --
- explain (costs off)
- select a.unique1, b.unique1, c.unique1, coalesce(b.twothousand, a.twothousand)
- from tenk1 a left join tenk1 b on b.thousand = a.unique1 left join tenk1 c on c.unique2 = coalesce(b.twothousand, a.twothousand)
- where a.unique2 < 10 and coalesce(b.twothousand, a.twothousand) = 44;
- select a.unique1, b.unique1, c.unique1, coalesce(b.twothousand, a.twothousand)
- from tenk1 a left join tenk1 b on b.thousand = a.unique1 left join tenk1 c on c.unique2 = coalesce(b.twothousand, a.twothousand)
- where a.unique2 < 10 and coalesce(b.twothousand, a.twothousand) = 44;
- --
- -- check handling of join aliases when flattening multiple levels of subquery
- --
- explain (verbose, costs off)
- select foo1.join_key as foo1_id, foo3.join_key AS foo3_id, bug_field from
- (values (0),(1)) foo1(join_key)
- left join
- (select join_key, bug_field from
- (select ss1.join_key, ss1.bug_field from
- (select f1 as join_key, 666 as bug_field from int4_tbl i1) ss1
- ) foo2
- left join
- (select unique2 as join_key from tenk1 i2) ss2
- using (join_key)
- ) foo3
- using (join_key);
- select foo1.join_key as foo1_id, foo3.join_key AS foo3_id, bug_field from
- (values (0),(1)) foo1(join_key)
- left join
- (select join_key, bug_field from
- (select ss1.join_key, ss1.bug_field from
- (select f1 as join_key, 666 as bug_field from int4_tbl i1) ss1
- ) foo2
- left join
- (select unique2 as join_key from tenk1 i2) ss2
- using (join_key)
- ) foo3
- using (join_key);
- --
- -- test successful handling of nested outer joins with degenerate join quals
- --
- explain (verbose, costs off)
- select t1.* from
- text_tbl t1
- left join (select *, '***'::text as d1 from int8_tbl i8b1) b1
- left join int8_tbl i8
- left join (select *, null::int as d2 from int8_tbl i8b2) b2
- on (i8.q1 = b2.q1)
- on (b2.d2 = b1.q2)
- on (t1.f1 = b1.d1)
- left join int4_tbl i4
- on (i8.q2 = i4.f1);
- select t1.* from
- text_tbl t1
- left join (select *, '***'::text as d1 from int8_tbl i8b1) b1
- left join int8_tbl i8
- left join (select *, null::int as d2 from int8_tbl i8b2) b2
- on (i8.q1 = b2.q1)
- on (b2.d2 = b1.q2)
- on (t1.f1 = b1.d1)
- left join int4_tbl i4
- on (i8.q2 = i4.f1);
- explain (verbose, costs off)
- select t1.* from
- text_tbl t1
- left join (select *, '***'::text as d1 from int8_tbl i8b1) b1
- left join int8_tbl i8
- left join (select *, null::int as d2 from int8_tbl i8b2, int4_tbl i4b2) b2
- on (i8.q1 = b2.q1)
- on (b2.d2 = b1.q2)
- on (t1.f1 = b1.d1)
- left join int4_tbl i4
- on (i8.q2 = i4.f1);
- select t1.* from
- text_tbl t1
- left join (select *, '***'::text as d1 from int8_tbl i8b1) b1
- left join int8_tbl i8
- left join (select *, null::int as d2 from int8_tbl i8b2, int4_tbl i4b2) b2
- on (i8.q1 = b2.q1)
- on (b2.d2 = b1.q2)
- on (t1.f1 = b1.d1)
- left join int4_tbl i4
- on (i8.q2 = i4.f1);
- explain (verbose, costs off)
- select t1.* from
- text_tbl t1
- left join (select *, '***'::text as d1 from int8_tbl i8b1) b1
- left join int8_tbl i8
- left join (select *, null::int as d2 from int8_tbl i8b2, int4_tbl i4b2
- where q1 = f1) b2
- on (i8.q1 = b2.q1)
- on (b2.d2 = b1.q2)
- on (t1.f1 = b1.d1)
- left join int4_tbl i4
- on (i8.q2 = i4.f1);
- select t1.* from
- text_tbl t1
- left join (select *, '***'::text as d1 from int8_tbl i8b1) b1
- left join int8_tbl i8
- left join (select *, null::int as d2 from int8_tbl i8b2, int4_tbl i4b2
- where q1 = f1) b2
- on (i8.q1 = b2.q1)
- on (b2.d2 = b1.q2)
- on (t1.f1 = b1.d1)
- left join int4_tbl i4
- on (i8.q2 = i4.f1);
- explain (verbose, costs off)
- select * from
- text_tbl t1
- inner join int8_tbl i8
- on i8.q2 = 456
- right join text_tbl t2
- on t1.f1 = 'doh!'
- left join int4_tbl i4
- on i8.q1 = i4.f1;
- select * from
- text_tbl t1
- inner join int8_tbl i8
- on i8.q2 = 456
- right join text_tbl t2
- on t1.f1 = 'doh!'
- left join int4_tbl i4
- on i8.q1 = i4.f1;
- --
- -- test for appropriate join order in the presence of lateral references
- --
- explain (verbose, costs off)
- select * from
- text_tbl t1
- left join int8_tbl i8
- on i8.q2 = 123,
- lateral (select i8.q1, t2.f1 from text_tbl t2 limit 1) as ss
- where t1.f1 = ss.f1;
- select * from
- text_tbl t1
- left join int8_tbl i8
- on i8.q2 = 123,
- lateral (select i8.q1, t2.f1 from text_tbl t2 limit 1) as ss
- where t1.f1 = ss.f1;
- explain (verbose, costs off)
- select * from
- text_tbl t1
- left join int8_tbl i8
- on i8.q2 = 123,
- lateral (select i8.q1, t2.f1 from text_tbl t2 limit 1) as ss1,
- lateral (select ss1.* from text_tbl t3 limit 1) as ss2
- where t1.f1 = ss2.f1;
- select * from
- text_tbl t1
- left join int8_tbl i8
- on i8.q2 = 123,
- lateral (select i8.q1, t2.f1 from text_tbl t2 limit 1) as ss1,
- lateral (select ss1.* from text_tbl t3 limit 1) as ss2
- where t1.f1 = ss2.f1;
- explain (verbose, costs off)
- select 1 from
- text_tbl as tt1
- inner join text_tbl as tt2 on (tt1.f1 = 'foo')
- left join text_tbl as tt3 on (tt3.f1 = 'foo')
- left join text_tbl as tt4 on (tt3.f1 = tt4.f1),
- lateral (select tt4.f1 as c0 from text_tbl as tt5 limit 1) as ss1
- where tt1.f1 = ss1.c0;
- select 1 from
- text_tbl as tt1
- inner join text_tbl as tt2 on (tt1.f1 = 'foo')
- left join text_tbl as tt3 on (tt3.f1 = 'foo')
- left join text_tbl as tt4 on (tt3.f1 = tt4.f1),
- lateral (select tt4.f1 as c0 from text_tbl as tt5 limit 1) as ss1
- where tt1.f1 = ss1.c0;
- --
- -- check a case in which a PlaceHolderVar forces join order
- --
- explain (verbose, costs off)
- select ss2.* from
- int4_tbl i41
- left join int8_tbl i8
- join (select i42.f1 as c1, i43.f1 as c2, 42 as c3
- from int4_tbl i42, int4_tbl i43) ss1
- on i8.q1 = ss1.c2
- on i41.f1 = ss1.c1,
- lateral (select i41.*, i8.*, ss1.* from text_tbl limit 1) ss2
- where ss1.c2 = 0;
- select ss2.* from
- int4_tbl i41
- left join int8_tbl i8
- join (select i42.f1 as c1, i43.f1 as c2, 42 as c3
- from int4_tbl i42, int4_tbl i43) ss1
- on i8.q1 = ss1.c2
- on i41.f1 = ss1.c1,
- lateral (select i41.*, i8.*, ss1.* from text_tbl limit 1) ss2
- where ss1.c2 = 0;
- --
- -- test successful handling of full join underneath left join (bug #14105)
- --
- explain (costs off)
- select * from
- (select 1 as id) as xx
- left join
- (tenk1 as a1 full join (select 1 as id) as yy on (a1.unique1 = yy.id))
- on (xx.id = coalesce(yy.id));
- select * from
- (select 1 as id) as xx
- left join
- (tenk1 as a1 full join (select 1 as id) as yy on (a1.unique1 = yy.id))
- on (xx.id = coalesce(yy.id));
- --
- -- test ability to push constants through outer join clauses
- --
- explain (costs off)
- select * from int4_tbl a left join tenk1 b on f1 = unique2 where f1 = 0;
- explain (costs off)
- select * from tenk1 a full join tenk1 b using(unique2) where unique2 = 42;
- --
- -- test that quals attached to an outer join have correct semantics,
- -- specifically that they don't re-use expressions computed below the join;
- -- we force a mergejoin so that coalesce(b.q1, 1) appears as a join input
- --
- set enable_hashjoin to off;
- set enable_nestloop to off;
- explain (verbose, costs off)
- select a.q2, b.q1
- from int8_tbl a left join int8_tbl b on a.q2 = coalesce(b.q1, 1)
- where coalesce(b.q1, 1) > 0;
- select a.q2, b.q1
- from int8_tbl a left join int8_tbl b on a.q2 = coalesce(b.q1, 1)
- where coalesce(b.q1, 1) > 0;
- reset enable_hashjoin;
- reset enable_nestloop;
- --
- -- test join removal
- --
- begin;
- CREATE TEMP TABLE a (id int PRIMARY KEY, b_id int);
- CREATE TEMP TABLE b (id int PRIMARY KEY, c_id int);
- CREATE TEMP TABLE c (id int PRIMARY KEY);
- CREATE TEMP TABLE d (a int, b int);
- INSERT INTO a VALUES (0, 0), (1, NULL);
- INSERT INTO b VALUES (0, 0), (1, NULL);
- INSERT INTO c VALUES (0), (1);
- INSERT INTO d VALUES (1,3), (2,2), (3,1);
- -- all three cases should be optimizable into a simple seqscan
- explain (costs off) SELECT a.* FROM a LEFT JOIN b ON a.b_id = b.id;
- explain (costs off) SELECT b.* FROM b LEFT JOIN c ON b.c_id = c.id;
- explain (costs off)
- SELECT a.* FROM a LEFT JOIN (b left join c on b.c_id = c.id)
- ON (a.b_id = b.id);
- -- check optimization of outer join within another special join
- explain (costs off)
- select id from a where id in (
- select b.id from b left join c on b.id = c.id
- );
- -- check that join removal works for a left join when joining a subquery
- -- that is guaranteed to be unique by its GROUP BY clause
- explain (costs off)
- select d.* from d left join (select * from b group by b.id, b.c_id) s
- on d.a = s.id and d.b = s.c_id;
- -- similarly, but keying off a DISTINCT clause
- explain (costs off)
- select d.* from d left join (select distinct * from b) s
- on d.a = s.id and d.b = s.c_id;
- -- join removal is not possible when the GROUP BY contains a column that is
- -- not in the join condition. (Note: as of 9.6, we notice that b.id is a
- -- primary key and so drop b.c_id from the GROUP BY of the resulting plan;
- -- but this happens too late for join removal in the outer plan level.)
- explain (costs off)
- select d.* from d left join (select * from b group by b.id, b.c_id) s
- on d.a = s.id;
- -- similarly, but keying off a DISTINCT clause
- explain (costs off)
- select d.* from d left join (select distinct * from b) s
- on d.a = s.id;
- -- check join removal works when uniqueness of the join condition is enforced
- -- by a UNION
- explain (costs off)
- select d.* from d left join (select id from a union select id from b) s
- on d.a = s.id;
- -- check join removal with a cross-type comparison operator
- explain (costs off)
- select i8.* from int8_tbl i8 left join (select f1 from int4_tbl group by f1) i4
- on i8.q1 = i4.f1;
- -- check join removal with lateral references
- explain (costs off)
- select 1 from (select a.id FROM a left join b on a.b_id = b.id) q,
- lateral generate_series(1, q.id) gs(i) where q.id = gs.i;
- rollback;
- create temp table parent (k int primary key, pd int);
- create temp table child (k int unique, cd int);
- insert into parent values (1, 10), (2, 20), (3, 30);
- insert into child values (1, 100), (4, 400);
- -- this case is optimizable
- select p.* from parent p left join child c on (p.k = c.k);
- explain (costs off)
- select p.* from parent p left join child c on (p.k = c.k);
- -- this case is not
- select p.*, linked from parent p
- left join (select c.*, true as linked from child c) as ss
- on (p.k = ss.k);
- explain (costs off)
- select p.*, linked from parent p
- left join (select c.*, true as linked from child c) as ss
- on (p.k = ss.k);
- -- check for a 9.0rc1 bug: join removal breaks pseudoconstant qual handling
- select p.* from
- parent p left join child c on (p.k = c.k)
- where p.k = 1 and p.k = 2;
- explain (costs off)
- select p.* from
- parent p left join child c on (p.k = c.k)
- where p.k = 1 and p.k = 2;
- select p.* from
- (parent p left join child c on (p.k = c.k)) join parent x on p.k = x.k
- where p.k = 1 and p.k = 2;
- explain (costs off)
- select p.* from
- (parent p left join child c on (p.k = c.k)) join parent x on p.k = x.k
- where p.k = 1 and p.k = 2;
- -- bug 5255: this is not optimizable by join removal
- begin;
- CREATE TEMP TABLE a (id int PRIMARY KEY);
- CREATE TEMP TABLE b (id int PRIMARY KEY, a_id int);
- INSERT INTO a VALUES (0), (1);
- INSERT INTO b VALUES (0, 0), (1, NULL);
- SELECT * FROM b LEFT JOIN a ON (b.a_id = a.id) WHERE (a.id IS NULL OR a.id > 0);
- SELECT b.* FROM b LEFT JOIN a ON (b.a_id = a.id) WHERE (a.id IS NULL OR a.id > 0);
- rollback;
- -- another join removal bug: this is not optimizable, either
- begin;
- create temp table innertab (id int8 primary key, dat1 int8);
- insert into innertab values(123, 42);
- SELECT * FROM
- (SELECT 1 AS x) ss1
- LEFT JOIN
- (SELECT q1, q2, COALESCE(dat1, q1) AS y
- FROM int8_tbl LEFT JOIN innertab ON q2 = id) ss2
- ON true;
- rollback;
- -- another join removal bug: we must clean up correctly when removing a PHV
- begin;
- create temp table uniquetbl (f1 text unique);
- explain (costs off)
- select t1.* from
- uniquetbl as t1
- left join (select *, '***'::text as d1 from uniquetbl) t2
- on t1.f1 = t2.f1
- left join uniquetbl t3
- on t2.d1 = t3.f1;
- explain (costs off)
- select t0.*
- from
- text_tbl t0
- left join
- (select case t1.ten when 0 then 'doh!'::text else null::text end as case1,
- t1.stringu2
- from tenk1 t1
- join int4_tbl i4 ON i4.f1 = t1.unique2
- left join uniquetbl u1 ON u1.f1 = t1.string4) ss
- on t0.f1 = ss.case1
- where ss.stringu2 !~* ss.case1;
- select t0.*
- from
- text_tbl t0
- left join
- (select case t1.ten when 0 then 'doh!'::text else null::text end as case1,
- t1.stringu2
- from tenk1 t1
- join int4_tbl i4 ON i4.f1 = t1.unique2
- left join uniquetbl u1 ON u1.f1 = t1.string4) ss
- on t0.f1 = ss.case1
- where ss.stringu2 !~* ss.case1;
- rollback;
- -- test case to expose miscomputation of required relid set for a PHV
- explain (verbose, costs off)
- select i8.*, ss.v, t.unique2
- from int8_tbl i8
- left join int4_tbl i4 on i4.f1 = 1
- left join lateral (select i4.f1 + 1 as v) as ss on true
- left join tenk1 t on t.unique2 = ss.v
- where q2 = 456;
- select i8.*, ss.v, t.unique2
- from int8_tbl i8
- left join int4_tbl i4 on i4.f1 = 1
- left join lateral (select i4.f1 + 1 as v) as ss on true
- left join tenk1 t on t.unique2 = ss.v
- where q2 = 456;
- -- and check a related issue where we miscompute required relids for
- -- a PHV that's been translated to a child rel
- create temp table parttbl (a integer primary key) partition by range (a);
- create temp table parttbl1 partition of parttbl for values from (1) to (100);
- insert into parttbl values (11), (12);
- explain (costs off)
- select * from
- (select *, 12 as phv from parttbl) as ss
- right join int4_tbl on true
- where ss.a = ss.phv and f1 = 0;
- select * from
- (select *, 12 as phv from parttbl) as ss
- right join int4_tbl on true
- where ss.a = ss.phv and f1 = 0;
- -- bug #8444: we've historically allowed duplicate aliases within aliased JOINs
- select * from
- int8_tbl x join (int4_tbl x cross join int4_tbl y) j on q1 = f1; -- error
- select * from
- int8_tbl x join (int4_tbl x cross join int4_tbl y) j on q1 = y.f1; -- error
- select * from
- int8_tbl x join (int4_tbl x cross join int4_tbl y(ff)) j on q1 = f1; -- ok
- --
- -- Test hints given on incorrect column references are useful
- --
- select t1.uunique1 from
- tenk1 t1 join tenk2 t2 on t1.two = t2.two; -- error, prefer "t1" suggestion
- select t2.uunique1 from
- tenk1 t1 join tenk2 t2 on t1.two = t2.two; -- error, prefer "t2" suggestion
- select uunique1 from
- tenk1 t1 join tenk2 t2 on t1.two = t2.two; -- error, suggest both at once
- --
- -- Take care to reference the correct RTE
- --
- select atts.relid::regclass, s.* from pg_stats s join
- pg_attribute a on s.attname = a.attname and s.tablename =
- a.attrelid::regclass::text join (select unnest(indkey) attnum,
- indexrelid from pg_index i) atts on atts.attnum = a.attnum where
- schemaname != 'pg_catalog';
- --
- -- Test LATERAL
- --
- select unique2, x.*
- from tenk1 a, lateral (select * from int4_tbl b where f1 = a.unique1) x;
- explain (costs off)
- select unique2, x.*
- from tenk1 a, lateral (select * from int4_tbl b where f1 = a.unique1) x;
- select unique2, x.*
- from int4_tbl x, lateral (select unique2 from tenk1 where f1 = unique1) ss;
- explain (costs off)
- select unique2, x.*
- from int4_tbl x, lateral (select unique2 from tenk1 where f1 = unique1) ss;
- explain (costs off)
- select unique2, x.*
- from int4_tbl x cross join lateral (select unique2 from tenk1 where f1 = unique1) ss;
- select unique2, x.*
- from int4_tbl x left join lateral (select unique1, unique2 from tenk1 where f1 = unique1) ss on true;
- explain (costs off)
- select unique2, x.*
- from int4_tbl x left join lateral (select unique1, unique2 from tenk1 where f1 = unique1) ss on true;
- -- check scoping of lateral versus parent references
- -- the first of these should return int8_tbl.q2, the second int8_tbl.q1
- select *, (select r from (select q1 as q2) x, (select q2 as r) y) from int8_tbl;
- select *, (select r from (select q1 as q2) x, lateral (select q2 as r) y) from int8_tbl;
- -- lateral with function in FROM
- select count(*) from tenk1 a, lateral generate_series(1,two) g;
- explain (costs off)
- select count(*) from tenk1 a, lateral generate_series(1,two) g;
- explain (costs off)
- select count(*) from tenk1 a cross join lateral generate_series(1,two) g;
- -- don't need the explicit LATERAL keyword for functions
- explain (costs off)
- select count(*) from tenk1 a, generate_series(1,two) g;
- -- lateral with UNION ALL subselect
- explain (costs off)
- select * from generate_series(100,200) g,
- lateral (select * from int8_tbl a where g = q1 union all
- select * from int8_tbl b where g = q2) ss;
- select * from generate_series(100,200) g,
- lateral (select * from int8_tbl a where g = q1 union all
- select * from int8_tbl b where g = q2) ss;
- -- lateral with VALUES
- explain (costs off)
- select count(*) from tenk1 a,
- tenk1 b join lateral (values(a.unique1)) ss(x) on b.unique2 = ss.x;
- select count(*) from tenk1 a,
- tenk1 b join lateral (values(a.unique1)) ss(x) on b.unique2 = ss.x;
- -- lateral with VALUES, no flattening possible
- explain (costs off)
- select count(*) from tenk1 a,
- tenk1 b join lateral (values(a.unique1),(-1)) ss(x) on b.unique2 = ss.x;
- select count(*) from tenk1 a,
- tenk1 b join lateral (values(a.unique1),(-1)) ss(x) on b.unique2 = ss.x;
- -- lateral injecting a strange outer join condition
- explain (costs off)
- select * from int8_tbl a,
- int8_tbl x left join lateral (select a.q1 from int4_tbl y) ss(z)
- on x.q2 = ss.z
- order by a.q1, a.q2, x.q1, x.q2, ss.z;
- select * from int8_tbl a,
- int8_tbl x left join lateral (select a.q1 from int4_tbl y) ss(z)
- on x.q2 = ss.z
- order by a.q1, a.q2, x.q1, x.q2, ss.z;
- -- lateral reference to a join alias variable
- select * from (select f1/2 as x from int4_tbl) ss1 join int4_tbl i4 on x = f1,
- lateral (select x) ss2(y);
- select * from (select f1 as x from int4_tbl) ss1 join int4_tbl i4 on x = f1,
- lateral (values(x)) ss2(y);
- select * from ((select f1/2 as x from int4_tbl) ss1 join int4_tbl i4 on x = f1) j,
- lateral (select x) ss2(y);
- -- lateral references requiring pullup
- select * from (values(1)) x(lb),
- lateral generate_series(lb,4) x4;
- select * from (select f1/1000000000 from int4_tbl) x(lb),
- lateral generate_series(lb,4) x4;
- select * from (values(1)) x(lb),
- lateral (values(lb)) y(lbcopy);
- select * from (values(1)) x(lb),
- lateral (select lb from int4_tbl) y(lbcopy);
- select * from
- int8_tbl x left join (select q1,coalesce(q2,0) q2 from int8_tbl) y on x.q2 = y.q1,
- lateral (values(x.q1,y.q1,y.q2)) v(xq1,yq1,yq2);
- select * from
- int8_tbl x left join (select q1,coalesce(q2,0) q2 from int8_tbl) y on x.q2 = y.q1,
- lateral (select x.q1,y.q1,y.q2) v(xq1,yq1,yq2);
- select x.* from
- int8_tbl x left join (select q1,coalesce(q2,0) q2 from int8_tbl) y on x.q2 = y.q1,
- lateral (select x.q1,y.q1,y.q2) v(xq1,yq1,yq2);
- select v.* from
- (int8_tbl x left join (select q1,coalesce(q2,0) q2 from int8_tbl) y on x.q2 = y.q1)
- left join int4_tbl z on z.f1 = x.q2,
- lateral (select x.q1,y.q1 union all select x.q2,y.q2) v(vx,vy);
- select v.* from
- (int8_tbl x left join (select q1,(select coalesce(q2,0)) q2 from int8_tbl) y on x.q2 = y.q1)
- left join int4_tbl z on z.f1 = x.q2,
- lateral (select x.q1,y.q1 union all select x.q2,y.q2) v(vx,vy);
- select v.* from
- (int8_tbl x left join (select q1,(select coalesce(q2,0)) q2 from int8_tbl) y on x.q2 = y.q1)
- left join int4_tbl z on z.f1 = x.q2,
- lateral (select x.q1,y.q1 from onerow union all select x.q2,y.q2 from onerow) v(vx,vy);
- explain (verbose, costs off)
- select * from
- int8_tbl a left join
- lateral (select *, a.q2 as x from int8_tbl b) ss on a.q2 = ss.q1;
- select * from
- int8_tbl a left join
- lateral (select *, a.q2 as x from int8_tbl b) ss on a.q2 = ss.q1;
- explain (verbose, costs off)
- select * from
- int8_tbl a left join
- lateral (select *, coalesce(a.q2, 42) as x from int8_tbl b) ss on a.q2 = ss.q1;
- select * from
- int8_tbl a left join
- lateral (select *, coalesce(a.q2, 42) as x from int8_tbl b) ss on a.q2 = ss.q1;
- -- lateral can result in join conditions appearing below their
- -- real semantic level
- explain (verbose, costs off)
- select * from int4_tbl i left join
- lateral (select * from int2_tbl j where i.f1 = j.f1) k on true;
- select * from int4_tbl i left join
- lateral (select * from int2_tbl j where i.f1 = j.f1) k on true;
- explain (verbose, costs off)
- select * from int4_tbl i left join
- lateral (select coalesce(i) from int2_tbl j where i.f1 = j.f1) k on true;
- select * from int4_tbl i left join
- lateral (select coalesce(i) from int2_tbl j where i.f1 = j.f1) k on true;
- explain (verbose, costs off)
- select * from int4_tbl a,
- lateral (
- select * from int4_tbl b left join int8_tbl c on (b.f1 = q1 and a.f1 = q2)
- ) ss;
- select * from int4_tbl a,
- lateral (
- select * from int4_tbl b left join int8_tbl c on (b.f1 = q1 and a.f1 = q2)
- ) ss;
- -- lateral reference in a PlaceHolderVar evaluated at join level
- explain (verbose, costs off)
- select * from
- int8_tbl a left join lateral
- (select b.q1 as bq1, c.q1 as cq1, least(a.q1,b.q1,c.q1) from
- int8_tbl b cross join int8_tbl c) ss
- on a.q2 = ss.bq1;
- select * from
- int8_tbl a left join lateral
- (select b.q1 as bq1, c.q1 as cq1, least(a.q1,b.q1,c.q1) from
- int8_tbl b cross join int8_tbl c) ss
- on a.q2 = ss.bq1;
- -- case requiring nested PlaceHolderVars
- explain (verbose, costs off)
- select * from
- int8_tbl c left join (
- int8_tbl a left join (select q1, coalesce(q2,42) as x from int8_tbl b) ss1
- on a.q2 = ss1.q1
- cross join
- lateral (select q1, coalesce(ss1.x,q2) as y from int8_tbl d) ss2
- ) on c.q2 = ss2.q1,
- lateral (select ss2.y offset 0) ss3;
- -- case that breaks the old ph_may_need optimization
- explain (verbose, costs off)
- select c.*,a.*,ss1.q1,ss2.q1,ss3.* from
- int8_tbl c left join (
- int8_tbl a left join
- (select q1, coalesce(q2,f1) as x from int8_tbl b, int4_tbl b2
- where q1 < f1) ss1
- on a.q2 = ss1.q1
- cross join
- lateral (select q1, coalesce(ss1.x,q2) as y from int8_tbl d) ss2
- ) on c.q2 = ss2.q1,
- lateral (select * from int4_tbl i where ss2.y > f1) ss3;
- -- check processing of postponed quals (bug #9041)
- explain (verbose, costs off)
- select * from
- (select 1 as x offset 0) x cross join (select 2 as y offset 0) y
- left join lateral (
- select * from (select 3 as z offset 0) z where z.z = x.x
- ) zz on zz.z = y.y;
- -- check dummy rels with lateral references (bug #15694)
- explain (verbose, costs off)
- select * from int8_tbl i8 left join lateral
- (select *, i8.q2 from int4_tbl where false) ss on true;
- explain (verbose, costs off)
- select * from int8_tbl i8 left join lateral
- (select *, i8.q2 from int4_tbl i1, int4_tbl i2 where false) ss on true;
- -- check handling of nested appendrels inside LATERAL
- select * from
- ((select 2 as v) union all (select 3 as v)) as q1
- cross join lateral
- ((select * from
- ((select 4 as v) union all (select 5 as v)) as q3)
- union all
- (select q1.v)
- ) as q2;
- -- check the number of columns specified
- SELECT * FROM (int8_tbl i cross join int4_tbl j) ss(a,b,c,d);
- -- check we don't try to do a unique-ified semijoin with LATERAL
- explain (verbose, costs off)
- select * from
- (values (0,9998), (1,1000)) v(id,x),
- lateral (select f1 from int4_tbl
- where f1 = any (select unique1 from tenk1
- where unique2 = v.x offset 0)) ss;
- select * from
- (values (0,9998), (1,1000)) v(id,x),
- lateral (select f1 from int4_tbl
- where f1 = any (select unique1 from tenk1
- where unique2 = v.x offset 0)) ss;
- -- check proper extParam/allParam handling (this isn't exactly a LATERAL issue,
- -- but we can make the test case much more compact with LATERAL)
- explain (verbose, costs off)
- select * from (values (0), (1)) v(id),
- lateral (select * from int8_tbl t1,
- lateral (select * from
- (select * from int8_tbl t2
- where q1 = any (select q2 from int8_tbl t3
- where q2 = (select greatest(t1.q1,t2.q2))
- and (select v.id=0)) offset 0) ss2) ss
- where t1.q1 = ss.q2) ss0;
- select * from (values (0), (1)) v(id),
- lateral (select * from int8_tbl t1,
- lateral (select * from
- (select * from int8_tbl t2
- where q1 = any (select q2 from int8_tbl t3
- where q2 = (select greatest(t1.q1,t2.q2))
- and (select v.id=0)) offset 0) ss2) ss
- where t1.q1 = ss.q2) ss0;
- -- test some error cases where LATERAL should have been used but wasn't
- select f1,g from int4_tbl a, (select f1 as g) ss;
- select f1,g from int4_tbl a, (select a.f1 as g) ss;
- select f1,g from int4_tbl a cross join (select f1 as g) ss;
- select f1,g from int4_tbl a cross join (select a.f1 as g) ss;
- -- SQL:2008 says the left table is in scope but illegal to access here
- select f1,g from int4_tbl a right join lateral generate_series(0, a.f1) g on true;
- select f1,g from int4_tbl a full join lateral generate_series(0, a.f1) g on true;
- -- check we complain about ambiguous table references
- select * from
- int8_tbl x cross join (int4_tbl x cross join lateral (select x.f1) ss);
- -- LATERAL can be used to put an aggregate into the FROM clause of its query
- select 1 from tenk1 a, lateral (select max(a.unique1) from int4_tbl b) ss;
- -- check behavior of LATERAL in UPDATE/DELETE
- create temp table xx1 as select f1 as x1, -f1 as x2 from int4_tbl;
- -- error, can't do this:
- update xx1 set x2 = f1 from (select * from int4_tbl where f1 = x1) ss;
- update xx1 set x2 = f1 from (select * from int4_tbl where f1 = xx1.x1) ss;
- -- can't do it even with LATERAL:
- update xx1 set x2 = f1 from lateral (select * from int4_tbl where f1 = x1) ss;
- -- we might in future allow something like this, but for now it's an error:
- update xx1 set x2 = f1 from xx1, lateral (select * from int4_tbl where f1 = x1) ss;
- -- also errors:
- delete from xx1 using (select * from int4_tbl where f1 = x1) ss;
- delete from xx1 using (select * from int4_tbl where f1 = xx1.x1) ss;
- delete from xx1 using lateral (select * from int4_tbl where f1 = x1) ss;
- --
- -- test LATERAL reference propagation down a multi-level inheritance hierarchy
- -- produced for a multi-level partitioned table hierarchy.
- --
- create table join_pt1 (a int, b int, c varchar) partition by range(a);
- create table join_pt1p1 partition of join_pt1 for values from (0) to (100) partition by range(b);
- create table join_pt1p2 partition of join_pt1 for values from (100) to (200);
- create table join_pt1p1p1 partition of join_pt1p1 for values from (0) to (100);
- insert into join_pt1 values (1, 1, 'x'), (101, 101, 'y');
- create table join_ut1 (a int, b int, c varchar);
- insert into join_ut1 values (101, 101, 'y'), (2, 2, 'z');
- explain (verbose, costs off)
- select t1.b, ss.phv from join_ut1 t1 left join lateral
- (select t2.a as t2a, t3.a t3a, least(t1.a, t2.a, t3.a) phv
- from join_pt1 t2 join join_ut1 t3 on t2.a = t3.b) ss
- on t1.a = ss.t2a order by t1.a;
- select t1.b, ss.phv from join_ut1 t1 left join lateral
- (select t2.a as t2a, t3.a t3a, least(t1.a, t2.a, t3.a) phv
- from join_pt1 t2 join join_ut1 t3 on t2.a = t3.b) ss
- on t1.a = ss.t2a order by t1.a;
- drop table join_pt1;
- drop table join_ut1;
- --
- -- test estimation behavior with multi-column foreign key and constant qual
- --
- begin;
- create table fkest (x integer, x10 integer, x10b integer, x100 integer);
- insert into fkest select x, x/10, x/10, x/100 from generate_series(1,1000) x;
- create unique index on fkest(x, x10, x100);
- analyze fkest;
- explain (costs off)
- select * from fkest f1
- join fkest f2 on (f1.x = f2.x and f1.x10 = f2.x10b and f1.x100 = f2.x100)
- join fkest f3 on f1.x = f3.x
- where f1.x100 = 2;
- alter table fkest add constraint fk
- foreign key (x, x10b, x100) references fkest (x, x10, x100);
- explain (costs off)
- select * from fkest f1
- join fkest f2 on (f1.x = f2.x and f1.x10 = f2.x10b and f1.x100 = f2.x100)
- join fkest f3 on f1.x = f3.x
- where f1.x100 = 2;
- rollback;
- --
- -- test that foreign key join estimation performs sanely for outer joins
- --
- begin;
- create table fkest (a int, b int, c int unique, primary key(a,b));
- create table fkest1 (a int, b int, primary key(a,b));
- insert into fkest select x/10, x%10, x from generate_series(1,1000) x;
- insert into fkest1 select x/10, x%10 from generate_series(1,1000) x;
- alter table fkest1
- add constraint fkest1_a_b_fkey foreign key (a,b) references fkest;
- analyze fkest;
- analyze fkest1;
- explain (costs off)
- select *
- from fkest f
- left join fkest1 f1 on f.a = f1.a and f.b = f1.b
- left join fkest1 f2 on f.a = f2.a and f.b = f2.b
- left join fkest1 f3 on f.a = f3.a and f.b = f3.b
- where f.c = 1;
- rollback;
- --
- -- test planner's ability to mark joins as unique
- --
- create table j1 (id int primary key);
- create table j2 (id int primary key);
- create table j3 (id int);
- insert into j1 values(1),(2),(3);
- insert into j2 values(1),(2),(3);
- insert into j3 values(1),(1);
- analyze j1;
- analyze j2;
- analyze j3;
- -- ensure join is properly marked as unique
- explain (verbose, costs off)
- select * from j1 inner join j2 on j1.id = j2.id;
- -- ensure join is not unique when not an equi-join
- explain (verbose, costs off)
- select * from j1 inner join j2 on j1.id > j2.id;
- -- ensure non-unique rel is not chosen as inner
- explain (verbose, costs off)
- select * from j1 inner join j3 on j1.id = j3.id;
- -- ensure left join is marked as unique
- explain (verbose, costs off)
- select * from j1 left join j2 on j1.id = j2.id;
- -- ensure right join is marked as unique
- explain (verbose, costs off)
- select * from j1 right join j2 on j1.id = j2.id;
- -- ensure full join is marked as unique
- explain (verbose, costs off)
- select * from j1 full join j2 on j1.id = j2.id;
- -- a clauseless (cross) join can't be unique
- explain (verbose, costs off)
- select * from j1 cross join j2;
- -- ensure a natural join is marked as unique
- explain (verbose, costs off)
- select * from j1 natural join j2;
- -- ensure a distinct clause allows the inner to become unique
- explain (verbose, costs off)
- select * from j1
- inner join (select distinct id from j3) j3 on j1.id = j3.id;
- -- ensure group by clause allows the inner to become unique
- explain (verbose, costs off)
- select * from j1
- inner join (select id from j3 group by id) j3 on j1.id = j3.id;
- drop table j1;
- drop table j2;
- drop table j3;
- -- test more complex permutations of unique joins
- create table j1 (id1 int, id2 int, primary key(id1,id2));
- create table j2 (id1 int, id2 int, primary key(id1,id2));
- create table j3 (id1 int, id2 int, primary key(id1,id2));
- insert into j1 values(1,1),(1,2);
- insert into j2 values(1,1);
- insert into j3 values(1,1);
- analyze j1;
- analyze j2;
- analyze j3;
- -- ensure there's no unique join when not all columns which are part of the
- -- unique index are seen in the join clause
- explain (verbose, costs off)
- select * from j1
- inner join j2 on j1.id1 = j2.id1;
- -- ensure proper unique detection with multiple join quals
- explain (verbose, costs off)
- select * from j1
- inner join j2 on j1.id1 = j2.id1 and j1.id2 = j2.id2;
- -- ensure we don't detect the join to be unique when quals are not part of the
- -- join condition
- explain (verbose, costs off)
- select * from j1
- inner join j2 on j1.id1 = j2.id1 where j1.id2 = 1;
- -- as above, but for left joins.
- explain (verbose, costs off)
- select * from j1
- left join j2 on j1.id1 = j2.id1 where j1.id2 = 1;
- -- validate logic in merge joins which skips mark and restore.
- -- it should only do this if all quals which were used to detect the unique
- -- are present as join quals, and not plain quals.
- set enable_nestloop to 0;
- set enable_hashjoin to 0;
- set enable_sort to 0;
- -- create indexes that will be preferred over the PKs to perform the join
- create index j1_id1_idx on j1 (id1) where id1 % 1000 = 1;
- create index j2_id1_idx on j2 (id1) where id1 % 1000 = 1;
- -- need an additional row in j2, if we want j2_id1_idx to be preferred
- insert into j2 values(1,2);
- analyze j2;
- explain (costs off) select * from j1
- inner join j2 on j1.id1 = j2.id1 and j1.id2 = j2.id2
- where j1.id1 % 1000 = 1 and j2.id1 % 1000 = 1;
- select * from j1
- inner join j2 on j1.id1 = j2.id1 and j1.id2 = j2.id2
- where j1.id1 % 1000 = 1 and j2.id1 % 1000 = 1;
- -- Exercise array keys mark/restore B-Tree code
- explain (costs off) select * from j1
- inner join j2 on j1.id1 = j2.id1 and j1.id2 = j2.id2
- where j1.id1 % 1000 = 1 and j2.id1 % 1000 = 1 and j2.id1 = any (array[1]);
- select * from j1
- inner join j2 on j1.id1 = j2.id1 and j1.id2 = j2.id2
- where j1.id1 % 1000 = 1 and j2.id1 % 1000 = 1 and j2.id1 = any (array[1]);
- -- Exercise array keys "find extreme element" B-Tree code
- explain (costs off) select * from j1
- inner join j2 on j1.id1 = j2.id1 and j1.id2 = j2.id2
- where j1.id1 % 1000 = 1 and j2.id1 % 1000 = 1 and j2.id1 >= any (array[1,5]);
- select * from j1
- inner join j2 on j1.id1 = j2.id1 and j1.id2 = j2.id2
- where j1.id1 % 1000 = 1 and j2.id1 % 1000 = 1 and j2.id1 >= any (array[1,5]);
- reset enable_nestloop;
- reset enable_hashjoin;
- reset enable_sort;
- drop table j1;
- drop table j2;
- drop table j3;
- -- check that semijoin inner is not seen as unique for a portion of the outerrel
- explain (verbose, costs off)
- select t1.unique1, t2.hundred
- from onek t1, tenk1 t2
- where exists (select 1 from tenk1 t3
- where t3.thousand = t1.unique1 and t3.tenthous = t2.hundred)
- and t1.unique1 < 1;
- -- ... unless it actually is unique
- create table j3 as select unique1, tenthous from onek;
- vacuum analyze j3;
- create unique index on j3(unique1, tenthous);
- explain (verbose, costs off)
- select t1.unique1, t2.hundred
- from onek t1, tenk1 t2
- where exists (select 1 from j3
- where j3.unique1 = t1.unique1 and j3.tenthous = t2.hundred)
- and t1.unique1 < 1;
- drop table j3;
|