mirror of
https://github.com/FirebirdSQL/firebird-qa.git
synced 2025-01-22 13:33:07 +01:00
268 lines
9.1 KiB
Python
268 lines
9.1 KiB
Python
#coding:utf-8
|
|
|
|
"""
|
|
ID: issue-7904
|
|
ISSUE: 7904
|
|
TITLE: More realistic cardinality adjustments for unmatchable booleans // FB5 bad plan for query
|
|
DESCRIPTION:
|
|
NOTES:
|
|
Confirmed problem on 5.0.0.1291 (for UMOWA_ROWS = 700K number of fetches = 6059386, elapsed time = 9.609s)
|
|
Checked on 5.0.0.1303, 6.0.0.180 (for UMOWA_ROWS = 700K number of fetches = 270208, elapsed time = 0.741s)
|
|
|
|
[24.09.2024] pzotov
|
|
Changed substitutions: one need to suppress '(keys: N, total key length: M)' in FB 6.x (and ONLY there),
|
|
otherwise actual and expected output become differ.
|
|
Commit: https://github.com/FirebirdSQL/firebird/commit/c50b0aa652014ce3610a1890017c9dd436388c43
|
|
("Add key info to the hash join plan output", 23.09.2024 18:26)
|
|
Discussed with dimitr.
|
|
Checked on 6.0.0.467-cc183f5, 5.0.2.1513
|
|
"""
|
|
|
|
import pytest
|
|
from firebird.qa import *
|
|
|
|
UMOWA_ROWS = 7000
|
|
ROZL_MULTIPLIER = 10
|
|
|
|
|
|
init_sql = f"""
|
|
set bail on;
|
|
|
|
create table umowa
|
|
(
|
|
umowa_id char(8) not null,
|
|
dyr_id smallint not null,
|
|
umowa_id_seq smallint not null,
|
|
typ_umowy_id char(1) not null,
|
|
rodz_umowy_id char(2) not null,
|
|
constraint
|
|
pk_umowa primary key (umowa_id,dyr_id,umowa_id_seq)
|
|
using index pk_umowa
|
|
);
|
|
|
|
create table dok_rozliczeniowy
|
|
(
|
|
dok_rozliczeniowy_id char(2) not null,
|
|
dok_rozliczeniowy_inkaso char(1) not null,
|
|
constraint
|
|
pk_dok_rozliczeniowy primary key (dok_rozliczeniowy_id)
|
|
using index pk_dok_rozliczeniowy
|
|
);
|
|
|
|
create table rozliczenie
|
|
(
|
|
dyr_id smallint not null,
|
|
insp_id smallint not null,
|
|
okres_numer char(7) not null,
|
|
rozlicz_nr smallint not null,
|
|
rozlicz_nr_poz smallint not null,
|
|
umowa_id char(8) not null,
|
|
umowa_id_seq smallint not null,
|
|
umowa_id_poz smallint not null,
|
|
dok_rozliczeniowy_id char(2) not null,
|
|
rozlicz_rodz_dzial_id char(3),
|
|
rozlicz_kwota_rozliczona decimal(10,2) not null,
|
|
constraint
|
|
pk_rozliczenie primary key (dyr_id,insp_id,okres_numer,rozlicz_nr,rozlicz_nr_poz)
|
|
using index pk_rozliczenie
|
|
);
|
|
|
|
create table rodzaj_umowy
|
|
(
|
|
rodz_umowy_id char(2) not null,
|
|
typ_umowy_id char(1) not null,
|
|
constraint
|
|
pk_rodzaj_umowy
|
|
primary key (rodz_umowy_id,typ_umowy_id)
|
|
using index pk_rodzaj_umowy
|
|
);
|
|
|
|
create table dyrekcja (
|
|
dyr_id smallint not null primary key using index pk_dyrekcja
|
|
);
|
|
|
|
|
|
set term ^ ;
|
|
recreate procedure fill_data
|
|
as
|
|
declare rozlicz_nr integer;
|
|
declare umowa_id integer;
|
|
declare dyr_id integer;
|
|
declare typ_umowy_id integer;
|
|
declare rodz_umowy_id integer;
|
|
declare umowa_id_seq integer;
|
|
declare var_i integer;
|
|
declare okres_numer integer;
|
|
begin
|
|
umowa_id = 1;
|
|
rozlicz_nr = 1;
|
|
dyr_id = 1;
|
|
typ_umowy_id = 1;
|
|
rodz_umowy_id = 1;
|
|
okres_numer = 1;
|
|
while (umowa_id < {UMOWA_ROWS}) do
|
|
begin
|
|
if ( mod(umowa_id, 100) < 95 ) then
|
|
umowa_id_seq = 0;
|
|
else
|
|
umowa_id_seq = 1;
|
|
|
|
-- primary key (rodz_umowy_id,typ_umowy_id)
|
|
update or insert into rodzaj_umowy (rodz_umowy_id, typ_umowy_id)
|
|
values (
|
|
:rodz_umowy_id,
|
|
:typ_umowy_id
|
|
);
|
|
|
|
-- pk_dok_rozliczeniowy primary key (dok_rozliczeniowy_id)
|
|
update or insert into dok_rozliczeniowy (dok_rozliczeniowy_id, dok_rozliczeniowy_inkaso)
|
|
values (
|
|
:rodz_umowy_id,
|
|
:typ_umowy_id
|
|
);
|
|
|
|
|
|
insert into umowa (umowa_id, dyr_id, umowa_id_seq, typ_umowy_id, rodz_umowy_id)
|
|
values (
|
|
:umowa_id,
|
|
:dyr_id,
|
|
:umowa_id_seq,
|
|
:typ_umowy_id,
|
|
:rodz_umowy_id
|
|
);
|
|
|
|
var_i = 1;
|
|
while (var_i < {ROZL_MULTIPLIER}) do
|
|
begin
|
|
insert into rozliczenie (dyr_id, insp_id, okres_numer, rozlicz_nr,
|
|
rozlicz_nr_poz, umowa_id, umowa_id_seq, umowa_id_poz, dok_rozliczeniowy_id,
|
|
rozlicz_rodz_dzial_id, rozlicz_kwota_rozliczona
|
|
) values (
|
|
:dyr_id, :rodz_umowy_id, :okres_numer, :rozlicz_nr,
|
|
:var_i, :umowa_id, :umowa_id_seq, :var_i, :rodz_umowy_id,
|
|
:rodz_umowy_id, 1
|
|
);
|
|
|
|
rozlicz_nr = rozlicz_nr + 1;
|
|
if (rozlicz_nr > 3000) then
|
|
begin
|
|
rozlicz_nr = 1;
|
|
okres_numer = okres_numer + 1;
|
|
end
|
|
var_i = var_i + 1;
|
|
end
|
|
|
|
umowa_id = umowa_id + 1;
|
|
dyr_id = dyr_id + 1;
|
|
typ_umowy_id = typ_umowy_id + 1;
|
|
rodz_umowy_id = rodz_umowy_id + 1;
|
|
if (dyr_id > 16) then
|
|
dyr_id = 1;
|
|
if (typ_umowy_id > 2) then
|
|
typ_umowy_id = 1;
|
|
if (rodz_umowy_id > 40) then
|
|
rodz_umowy_id = 1;
|
|
end
|
|
end
|
|
^
|
|
set term ;^
|
|
commit;
|
|
|
|
execute procedure fill_data;
|
|
|
|
insert into dyrekcja(dyr_id)
|
|
select distinct dyr_id from rozliczenie;
|
|
commit;
|
|
|
|
alter table rozliczenie add constraint fk_rozliczenie__umowa foreign key(umowa_id, dyr_id, umowa_id_seq) references umowa(umowa_id, dyr_id, umowa_id_seq) on update cascade;
|
|
alter table umowa add constraint fk_umowa__rodzaj_umowy foreign key(rodz_umowy_id, typ_umowy_id) references rodzaj_umowy(rodz_umowy_id, typ_umowy_id) on update cascade;
|
|
alter table rozliczenie add constraint rozliczenie_fk4 foreign key(dok_rozliczeniowy_id) references dok_rozliczeniowy(dok_rozliczeniowy_id);
|
|
alter table rozliczenie add constraint fk_rozliczenie__dyrekcja foreign key (dyr_id) references dyrekcja (dyr_id);
|
|
|
|
set statistics index pk_umowa;
|
|
set statistics index pk_dok_rozliczeniowy;
|
|
set statistics index pk_rozliczenie;
|
|
set statistics index pk_rodzaj_umowy;
|
|
set statistics index pk_dyrekcja;
|
|
commit;
|
|
|
|
"""
|
|
#-----------------------------------------------------------
|
|
|
|
db = db_factory(init = init_sql)
|
|
|
|
substitutions = \
|
|
[
|
|
( r'\(record length: \d+, key length: \d+\)', '' ) # (record length: 132, key length: 16)
|
|
,( r'\(keys: \d+, total key length: \d+\)', '' ) # (keys: 1, total key length: 2)
|
|
]
|
|
|
|
act = python_act('db', substitutions = substitutions)
|
|
|
|
#-----------------------------------------------------------
|
|
|
|
query_lst = [
|
|
# Query from https://github.com/FirebirdSQL/firebird/issues/7904:
|
|
"""
|
|
select
|
|
q1_rozl.dyr_id
|
|
, q1_rozl.rozlicz_rodz_dzial_id
|
|
, sum(q1_rozl.rozlicz_kwota_rozliczona)
|
|
from
|
|
rozliczenie q1_rozl
|
|
inner join dok_rozliczeniowy q1_dokr
|
|
on q1_dokr.dok_rozliczeniowy_id = q1_rozl.dok_rozliczeniowy_id
|
|
inner join umowa q1_umowa
|
|
on q1_rozl.umowa_id = q1_umowa.umowa_id
|
|
and q1_rozl.dyr_id = q1_umowa.dyr_id
|
|
and q1_rozl.umowa_id_seq = q1_umowa.umowa_id_seq
|
|
where
|
|
q1_rozl.okres_numer between '15'
|
|
and '18'
|
|
and q1_dokr.dok_rozliczeniowy_inkaso = '1'
|
|
and q1_umowa.rodz_umowy_id='27'
|
|
group by
|
|
q1_rozl.dyr_id
|
|
, q1_rozl.rozlicz_rodz_dzial_id
|
|
""",
|
|
]
|
|
|
|
#---------------------------------------------------------
|
|
def replace_leading(source, char="."):
|
|
stripped = source.lstrip()
|
|
return char * (len(source) - len(stripped)) + stripped
|
|
#---------------------------------------------------------
|
|
|
|
@pytest.mark.version('>=5.0')
|
|
def test_1(act: Action, capsys):
|
|
with act.db.connect() as con:
|
|
cur = con.cursor()
|
|
for q in query_lst:
|
|
with cur.prepare(q) as ps:
|
|
print( '\n'.join([replace_leading(s) for s in ps.detailed_plan .split('\n')]) )
|
|
|
|
expected_stdout = """
|
|
Select Expression
|
|
....-> Aggregate
|
|
........-> Sort (record length: 132, key length: 16)
|
|
............-> Filter
|
|
................-> Hash Join (inner)
|
|
....................-> Nested Loop Join (inner)
|
|
........................-> Filter
|
|
............................-> Table "UMOWA" as "Q1_UMOWA" Access By ID
|
|
................................-> Bitmap
|
|
....................................-> Index "FK_UMOWA__RODZAJ_UMOWY" Range Scan (partial match: 1/2)
|
|
........................-> Filter
|
|
............................-> Table "ROZLICZENIE" as "Q1_ROZL" Access By ID
|
|
................................-> Bitmap
|
|
....................................-> Index "FK_ROZLICZENIE__UMOWA" Range Scan (full match)
|
|
....................-> Record Buffer (record length: 25)
|
|
........................-> Filter
|
|
............................-> Table "DOK_ROZLICZENIOWY" as "Q1_DOKR" Full Scan
|
|
"""
|
|
|
|
|
|
act.expected_stdout = expected_stdout
|
|
act.stdout = capsys.readouterr().out
|
|
assert act.clean_stdout == act.clean_expected_stdout
|