mirror of
https://github.com/FirebirdSQL/firebird-qa.git
synced 2025-01-22 13:33:07 +01:00
More python tests and some enhancements
This commit is contained in:
parent
76b49666fa
commit
0711661a93
BIN
files/core_5207.zip
Normal file
BIN
files/core_5207.zip
Normal file
Binary file not shown.
BIN
files/core_5579_broken_nn.zip
Normal file
BIN
files/core_5579_broken_nn.zip
Normal file
Binary file not shown.
BIN
files/core_5618.zip
Normal file
BIN
files/core_5618.zip
Normal file
Binary file not shown.
BIN
files/core_5637.zip
Normal file
BIN
files/core_5637.zip
Normal file
Binary file not shown.
BIN
files/core_5659.zip
Normal file
BIN
files/core_5659.zip
Normal file
Binary file not shown.
BIN
files/core_5719-ods-11_2.zip
Normal file
BIN
files/core_5719-ods-11_2.zip
Normal file
Binary file not shown.
@ -25,7 +25,7 @@
|
|||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, python_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
from firebird.driver import DbWriteMode, SrvRestoreFlag
|
from firebird.driver import SrvRestoreFlag
|
||||||
#from difflib import unified_diff
|
#from difflib import unified_diff
|
||||||
from io import BytesIO
|
from io import BytesIO
|
||||||
|
|
||||||
@ -521,8 +521,7 @@ act_1 = python_act('db_1', substitutions=substitutions_1)
|
|||||||
@pytest.mark.version('>=4.0')
|
@pytest.mark.version('>=4.0')
|
||||||
def test_1(act_1: Action):
|
def test_1(act_1: Action):
|
||||||
# CHANGE FW to OFF
|
# CHANGE FW to OFF
|
||||||
with act_1.connect_server() as srv:
|
act_1.db.set_async_write()
|
||||||
srv.database.set_write_mode(database=act_1.db.db_path, mode=DbWriteMode.ASYNC)
|
|
||||||
# 1. FIRST RUN DML_TEST
|
# 1. FIRST RUN DML_TEST
|
||||||
act_1.script = test_script_1
|
act_1.script = test_script_1
|
||||||
act_1.execute()
|
act_1.execute()
|
||||||
|
@ -30,7 +30,7 @@
|
|||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, python_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
from firebird.driver import DbWriteMode, DbInfoCode
|
from firebird.driver import DbInfoCode
|
||||||
|
|
||||||
# version: 2.5
|
# version: 2.5
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -172,8 +172,7 @@ act_1 = python_act('db_1', substitutions=substitutions_1)
|
|||||||
@pytest.mark.version('>=2.5')
|
@pytest.mark.version('>=2.5')
|
||||||
def test_1(act_1: Action):
|
def test_1(act_1: Action):
|
||||||
# Change FW to OFF in order to speed up initial data filling:
|
# Change FW to OFF in order to speed up initial data filling:
|
||||||
with act_1.connect_server() as srv:
|
act_1.db.set_async_write()
|
||||||
srv.database.set_write_mode(database=act_1.db.db_path, mode=DbWriteMode.ASYNC)
|
|
||||||
# prepare DB for testing: create lot of tables:
|
# prepare DB for testing: create lot of tables:
|
||||||
num_of_tables = 1000
|
num_of_tables = 1000
|
||||||
sql_ddl = f'''
|
sql_ddl = f'''
|
||||||
|
@ -279,6 +279,3 @@ def test_1(act_1: Action):
|
|||||||
act_1.expected_stdout = expected_stdout_1
|
act_1.expected_stdout = expected_stdout_1
|
||||||
act_1.trace_to_stdout()
|
act_1.trace_to_stdout()
|
||||||
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -42,7 +42,6 @@
|
|||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, python_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
from firebird.driver import DbWriteMode
|
|
||||||
|
|
||||||
# version: 2.5.2
|
# version: 2.5.2
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -458,8 +457,7 @@ trace_1 = ['log_transactions = true',
|
|||||||
def test_1(act_1: Action, capsys):
|
def test_1(act_1: Action, capsys):
|
||||||
NUM_ROWS_TO_BE_ADDED = 45000
|
NUM_ROWS_TO_BE_ADDED = 45000
|
||||||
# Change FW to OFF in order to speed up initial data filling
|
# Change FW to OFF in order to speed up initial data filling
|
||||||
with act_1.connect_server() as srv:
|
act_1.db.set_async_write()
|
||||||
srv.database.set_write_mode(database=act_1.db.db_path, mode=DbWriteMode.ASYNC)
|
|
||||||
# Make initial data filling into PERMANENT table for retrieving later number of data pages
|
# Make initial data filling into PERMANENT table for retrieving later number of data pages
|
||||||
# (it should be the same for any kind of tables, including GTTs):
|
# (it should be the same for any kind of tables, including GTTs):
|
||||||
with act_1.db.connect() as con:
|
with act_1.db.connect() as con:
|
||||||
|
@ -56,7 +56,7 @@ import re
|
|||||||
import subprocess
|
import subprocess
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
from firebird.qa import db_factory, python_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
from firebird.driver import DbWriteMode, ShutdownMethod, ShutdownMode
|
from firebird.driver import ShutdownMethod, ShutdownMode
|
||||||
|
|
||||||
# version: 3.0
|
# version: 3.0
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -572,8 +572,7 @@ def test_1(act_1: Action, capsys):
|
|||||||
"""
|
"""
|
||||||
act_1.isql(switches=[], input=sql_ddl)
|
act_1.isql(switches=[], input=sql_ddl)
|
||||||
# Temporay change FW to OFF in order to make DML faster:
|
# Temporay change FW to OFF in order to make DML faster:
|
||||||
with act_1.connect_server() as srv:
|
act_1.db.set_async_write()
|
||||||
srv.database.set_write_mode(database=act_1.db.db_path, mode=DbWriteMode.ASYNC)
|
|
||||||
#
|
#
|
||||||
sql_data = f"""
|
sql_data = f"""
|
||||||
set term ^;
|
set term ^;
|
||||||
@ -605,8 +604,7 @@ def test_1(act_1: Action, capsys):
|
|||||||
act_1.reset()
|
act_1.reset()
|
||||||
act_1.isql(switches=['-nod'], input=sql_data)
|
act_1.isql(switches=['-nod'], input=sql_data)
|
||||||
# Restore FW to ON (make sweep to do its work "harder"):
|
# Restore FW to ON (make sweep to do its work "harder"):
|
||||||
with act_1.connect_server() as srv:
|
act_1.db.set_async_write()
|
||||||
srv.database.set_write_mode(database=act_1.db.db_path, mode=DbWriteMode.SYNC)
|
|
||||||
# Trace
|
# Trace
|
||||||
with act_1.trace(db_events=trace_1):
|
with act_1.trace(db_events=trace_1):
|
||||||
# Traced action
|
# Traced action
|
||||||
|
@ -373,5 +373,5 @@ expected_stdout_1 = """
|
|||||||
|
|
||||||
@pytest.mark.version('>=4.0')
|
@pytest.mark.version('>=4.0')
|
||||||
def test_1(db_1):
|
def test_1(db_1):
|
||||||
pytest.skip("Test requires 3rd party encryption plugin")
|
pytest.skip("Requires encryption plugin")
|
||||||
#pytest.fail("Test not IMPLEMENTED")
|
#pytest.fail("Test not IMPLEMENTED")
|
||||||
|
@ -33,7 +33,6 @@ import subprocess
|
|||||||
import time
|
import time
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from firebird.qa import db_factory, python_act, Action, temp_file
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
from firebird.driver import DbWriteMode
|
|
||||||
|
|
||||||
# version: 3.0
|
# version: 3.0
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -254,8 +253,7 @@ heavy_output_1 = temp_file('heavy_script.out')
|
|||||||
@pytest.mark.version('>=3.0')
|
@pytest.mark.version('>=3.0')
|
||||||
def test_1(act_1: Action, heavy_script_1: Path, heavy_output_1: Path, capsys):
|
def test_1(act_1: Action, heavy_script_1: Path, heavy_output_1: Path, capsys):
|
||||||
# Change database FW to OFF in order to increase speed of insertions and output its header info
|
# Change database FW to OFF in order to increase speed of insertions and output its header info
|
||||||
with act_1.connect_server() as srv:
|
act_1.db.set_async_write()
|
||||||
srv.database.set_write_mode(database=act_1.db.db_path, mode=DbWriteMode.ASYNC)
|
|
||||||
# Preparing script for ISQL that will do 'heavy DML'
|
# Preparing script for ISQL that will do 'heavy DML'
|
||||||
heavy_script_1.write_text("""
|
heavy_script_1.write_text("""
|
||||||
recreate sequence g;
|
recreate sequence g;
|
||||||
|
@ -34,7 +34,6 @@
|
|||||||
import pytest
|
import pytest
|
||||||
from zipfile import Path
|
from zipfile import Path
|
||||||
from firebird.qa import db_factory, python_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
from firebird.driver import DbWriteMode
|
|
||||||
|
|
||||||
# version: 3.0
|
# version: 3.0
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -119,8 +118,7 @@ expected_stdout_1 = """
|
|||||||
|
|
||||||
@pytest.mark.version('>=3.0')
|
@pytest.mark.version('>=3.0')
|
||||||
def test_1(act_1: Action):
|
def test_1(act_1: Action):
|
||||||
with act_1.connect_server() as srv:
|
act_1.db.set_async_write()
|
||||||
srv.database.set_write_mode(database=act_1.db.db_path, mode=DbWriteMode.ASYNC)
|
|
||||||
# Read FNC scripts from zip file and execute it
|
# Read FNC scripts from zip file and execute it
|
||||||
script_file = Path(act_1.vars['files'] / 'core_4880.zip',
|
script_file = Path(act_1.vars['files'] / 'core_4880.zip',
|
||||||
at='core_4880_fnc.tmp')
|
at='core_4880_fnc.tmp')
|
||||||
|
@ -174,5 +174,5 @@ expected_stdout_1 = """
|
|||||||
|
|
||||||
@pytest.mark.version('>=3.0')
|
@pytest.mark.version('>=3.0')
|
||||||
def test_1(db_1):
|
def test_1(db_1):
|
||||||
pytest.skip("Test depends on 3rd party encryption plugin")
|
pytest.skip("Requires encryption plugin")
|
||||||
#pytest.fail("Test not IMPLEMENTED")
|
#pytest.fail("Test not IMPLEMENTED")
|
||||||
|
@ -26,7 +26,7 @@
|
|||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, python_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
from firebird.driver import DbWriteMode, TPB, Isolation
|
from firebird.driver import TPB, Isolation
|
||||||
|
|
||||||
# version: 2.5.6
|
# version: 2.5.6
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -85,8 +85,7 @@ expected_stdout_1 = """
|
|||||||
|
|
||||||
@pytest.mark.version('>=2.5.6')
|
@pytest.mark.version('>=2.5.6')
|
||||||
def test_1(act_1: Action):
|
def test_1(act_1: Action):
|
||||||
with act_1.connect_server() as srv:
|
act_1.db.set_async_write()
|
||||||
srv.database.set_write_mode(database=act_1.db.db_path, mode=DbWriteMode.ASYNC)
|
|
||||||
#
|
#
|
||||||
custom_tpb = TPB(isolation=Isolation.CONCURRENCY).get_buffer()
|
custom_tpb = TPB(isolation=Isolation.CONCURRENCY).get_buffer()
|
||||||
with act_1.db.connect(no_gc=True) as con:
|
with act_1.db.connect(no_gc=True) as con:
|
||||||
|
@ -21,7 +21,6 @@
|
|||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, python_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
from firebird.driver import DbWriteMode
|
|
||||||
|
|
||||||
# version: 4.0
|
# version: 4.0
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -126,8 +125,7 @@ test_sript_1 = """
|
|||||||
|
|
||||||
@pytest.mark.version('>=4.0')
|
@pytest.mark.version('>=4.0')
|
||||||
def test_1(act_1: Action):
|
def test_1(act_1: Action):
|
||||||
with act_1.connect_server() as srv:
|
act_1.db.set_async_write()
|
||||||
srv.database.set_write_mode(database=act_1.db.db_path, mode=DbWriteMode.ASYNC)
|
|
||||||
act_1.expected_stdout = expected_stdout_1
|
act_1.expected_stdout = expected_stdout_1
|
||||||
act_1.isql(switches=[], input=test_sript_1)
|
act_1.isql(switches=[], input=test_sript_1)
|
||||||
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -26,7 +26,6 @@
|
|||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, python_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
from firebird.driver import DbWriteMode
|
|
||||||
|
|
||||||
# version: 2.5.7
|
# version: 2.5.7
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -267,8 +266,7 @@ test_script_1 = f"""
|
|||||||
def test_1(act_1: Action):
|
def test_1(act_1: Action):
|
||||||
if act_1.get_server_architecture() == 'SS':
|
if act_1.get_server_architecture() == 'SS':
|
||||||
# Bucgcheck is reproduced on 2.5.7.27030 only when FW = OFF
|
# Bucgcheck is reproduced on 2.5.7.27030 only when FW = OFF
|
||||||
with act_1.connect_server() as srv:
|
act_1.db.set_async_write()
|
||||||
srv.database.set_write_mode(database=act_1.db.db_path, mode=DbWriteMode.ASYNC)
|
|
||||||
# Test
|
# Test
|
||||||
act_1.expected_stdout = expected_stdout_1
|
act_1.expected_stdout = expected_stdout_1
|
||||||
act_1.isql(switches=[], input=test_script_1)
|
act_1.isql(switches=[], input=test_script_1)
|
||||||
|
@ -33,7 +33,7 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 3.0.2
|
# version: 3.0.2
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -304,16 +304,88 @@ db_1 = db_factory(page_size=8192, sql_dialect=3, init=init_script_1)
|
|||||||
# cleanup( (f_trc_cfg, f_trc_lst, f_trc_log, f_trc_err, sql_log, sql_err, sql_cmd) )
|
# cleanup( (f_trc_cfg, f_trc_lst, f_trc_log, f_trc_err, sql_log, sql_err, sql_cmd) )
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
PLAN (TEST ORDER TEST_F01_ID)
|
PLAN (TEST ORDER TEST_F01_ID)
|
||||||
Number of fetches: acceptable.
|
Number of fetches: acceptable.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
FETCHES_THRESHOLD = 80
|
||||||
|
|
||||||
|
init_sql_1 = """
|
||||||
|
recreate table test
|
||||||
|
(
|
||||||
|
id int not null,
|
||||||
|
f01 int,
|
||||||
|
f02 int
|
||||||
|
);
|
||||||
|
|
||||||
|
set term ^;
|
||||||
|
create or alter procedure sp_add_init_data(a_rows_to_add int)
|
||||||
|
as
|
||||||
|
declare n int;
|
||||||
|
declare i int = 0;
|
||||||
|
begin
|
||||||
|
n = a_rows_to_add;
|
||||||
|
while (i < n) do
|
||||||
|
begin
|
||||||
|
insert into test(id, f01, f02) values(:i, nullif(mod(:i, :n/20), 0), iif(mod(:i,3)<2, 0, 1))
|
||||||
|
returning :i+1 into i;
|
||||||
|
end
|
||||||
|
end
|
||||||
|
^
|
||||||
|
set term ^;
|
||||||
|
commit;
|
||||||
|
|
||||||
|
execute procedure sp_add_init_data(300000);
|
||||||
|
commit;
|
||||||
|
|
||||||
|
create index test_f01_id on test(f01, id);
|
||||||
|
create index test_f02_only on test(f02);
|
||||||
|
commit;
|
||||||
|
"""
|
||||||
|
|
||||||
|
test_script_1 = """
|
||||||
|
set list on;
|
||||||
|
select count(*) cnt_check
|
||||||
|
from (
|
||||||
|
select *
|
||||||
|
from test
|
||||||
|
where f01 -- ###################################################################
|
||||||
|
IS NULL -- <<< ::: NB ::: we check here 'f01 is NULL', exactly as ticket says.
|
||||||
|
and f02=0 -- ###################################################################
|
||||||
|
order by f01, id
|
||||||
|
) ;
|
||||||
|
"""
|
||||||
|
|
||||||
|
trace_1 = ['time_threshold = 0',
|
||||||
|
'log_statement_finish = true',
|
||||||
|
'print_plan = true',
|
||||||
|
'print_perf = true',
|
||||||
|
'log_initfini = false',
|
||||||
|
]
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.2')
|
@pytest.mark.version('>=3.0.2')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action):
|
||||||
def test_1(db_1):
|
act_1.db.set_async_write()
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
act_1.isql(switches=[], input=init_sql_1)
|
||||||
|
#
|
||||||
|
with act_1.trace(db_events=trace_1):
|
||||||
|
act_1.reset()
|
||||||
|
act_1.isql(switches=[], input=test_script_1)
|
||||||
|
# Process trace
|
||||||
|
run_with_plan = ''
|
||||||
|
num_of_fetches = 99999999
|
||||||
|
for line in act_1.trace_log:
|
||||||
|
if line.lower().startswith('plan ('):
|
||||||
|
run_with_plan = line.strip().upper()
|
||||||
|
elif 'fetch(es)' in line:
|
||||||
|
words = line.split()
|
||||||
|
for k in range(len(words)):
|
||||||
|
if words[k].startswith('fetch'):
|
||||||
|
num_of_fetches = int(words[k-1])
|
||||||
|
# Check
|
||||||
|
assert run_with_plan == 'PLAN (TEST ORDER TEST_F01_ID)'
|
||||||
|
assert num_of_fetches < FETCHES_THRESHOLD
|
||||||
|
@ -14,13 +14,16 @@
|
|||||||
#
|
#
|
||||||
# 03-mar-2021: replaced 'xnet' with 'localhost' in order have ability to run this test on Linux.
|
# 03-mar-2021: replaced 'xnet' with 'localhost' in order have ability to run this test on Linux.
|
||||||
#
|
#
|
||||||
|
# [pcisar] 8.12.2021
|
||||||
|
# Fails with "no permission for remote access to database security.db" on Linux FB 4.0
|
||||||
|
#
|
||||||
# tracker_id: CORE-5496
|
# tracker_id: CORE-5496
|
||||||
# min_versions: ['3.0.2']
|
# min_versions: ['3.0.2']
|
||||||
# versions: 3.0.2
|
# versions: 3.0.2
|
||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 3.0.2
|
# version: 3.0.2
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -68,7 +71,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
SEC$USER_NAME SYSDBA
|
SEC$USER_NAME SYSDBA
|
||||||
@ -76,9 +80,33 @@ expected_stdout_1 = """
|
|||||||
Records affected: 1
|
Records affected: 1
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
test_script_1 = """
|
||||||
|
connect 'localhost:security.db';
|
||||||
|
create or alter user foo password '123' grant admin role using plugin Srp;
|
||||||
|
create or alter user rio password '123' grant admin role using plugin Srp;
|
||||||
|
create or alter user bar password '123' grant admin role using plugin Srp;
|
||||||
|
commit;
|
||||||
|
grant rdb$admin to sysdba granted by foo;
|
||||||
|
grant rdb$admin to sysdba granted by rio;
|
||||||
|
grant rdb$admin to sysdba granted by bar;
|
||||||
|
commit;
|
||||||
|
set list on;
|
||||||
|
set count on;
|
||||||
|
select sec$user_name, sec$plugin from sec$users where upper(sec$user_name) = upper('sysdba') and upper(sec$plugin) = upper('srp');
|
||||||
|
commit;
|
||||||
|
|
||||||
|
drop user foo using plugin Srp;
|
||||||
|
drop user rio using plugin Srp;
|
||||||
|
drop user bar using plugin Srp;
|
||||||
|
commit;
|
||||||
|
quit;
|
||||||
|
"""
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.2')
|
@pytest.mark.version('>=3.0.2')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action):
|
||||||
def test_1(db_1):
|
pytest.skip("Requires remote access to security.db")
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.isql(switches=['-q', '-b'], input=test_script_1)
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
|
||||||
|
|
||||||
|
@ -32,18 +32,33 @@
|
|||||||
#
|
#
|
||||||
# Checked on 3.0.2.32702 (CS/SC/SS), 4.0.0.563 (CS/SC/SS)
|
# Checked on 3.0.2.32702 (CS/SC/SS), 4.0.0.563 (CS/SC/SS)
|
||||||
#
|
#
|
||||||
|
# [pcisar] 8.12.2021
|
||||||
|
# Reimplementation does not work as expected on Linux 4.0
|
||||||
|
# gstat output:
|
||||||
|
# Data pages: total 97, encrypted 0, non-crypted 97
|
||||||
|
# Index pages: total 85, encrypted 0, non-crypted 85
|
||||||
|
# Blob pages: total 199, encrypted 0, non-crypted 199
|
||||||
|
# Generator pages: total 1, encrypted 0, non-crypted 1
|
||||||
|
# Validation does not report BLOB page errors, only data and index corruptions.
|
||||||
|
#
|
||||||
# tracker_id: CORE-5501
|
# tracker_id: CORE-5501
|
||||||
# min_versions: ['3.0.2']
|
# min_versions: ['3.0.2']
|
||||||
# versions: 3.0.2
|
# versions: 3.0.2
|
||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
|
from __future__ import annotations
|
||||||
|
from typing import Dict
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
import re
|
||||||
|
from struct import unpack_from
|
||||||
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
from firebird.driver import Connection
|
||||||
|
|
||||||
# version: 3.0.2
|
# version: 3.0.2
|
||||||
# resources: None
|
# resources: None
|
||||||
|
|
||||||
substitutions_1 = [('total \\d+,', 'total'), ('non-crypted \\d+', 'non-crypted'), ('crypted \\d+', 'crypted')]
|
substitutions_1 = [('total \\d+,', 'total'),
|
||||||
|
('non-crypted \\d+', 'non-crypted'), ('crypted \\d+', 'crypted')]
|
||||||
|
|
||||||
init_script_1 = """
|
init_script_1 = """
|
||||||
alter database drop linger;
|
alter database drop linger;
|
||||||
@ -454,19 +469,250 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
Data pages: total 63, encrypted 0, non-crypted 63
|
Data pages: total 63, encrypted 0, non-crypted 63
|
||||||
Index pages: total 88, encrypted 0, non-crypted 88
|
Index pages: total 88, encrypted 0, non-crypted 88
|
||||||
Blob pages: total 199, encrypted 0, non-crypted 199
|
Blob pages: total 199, encrypted 0, non-crypted 199
|
||||||
Other pages: total 115, ENCRYPTED 3 (DB problem!), non-crypted 112
|
Other pages: total 115, ENCRYPTED 3 (DB problem!), non-crypted 112
|
||||||
Detect all THREE page types with problem ? => YES
|
Detected all THREE page types with problem => YES
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
PAGE_TYPES = {0: "undef/free",
|
||||||
|
1: "DB header",
|
||||||
|
2: "PIP",
|
||||||
|
3: "TIP",
|
||||||
|
4: "Pntr Page",
|
||||||
|
5: "Data Page",
|
||||||
|
6: "Indx Root",
|
||||||
|
7: "Indx Data",
|
||||||
|
8: "Blob Page",
|
||||||
|
9: "Gens Page",
|
||||||
|
10: "SCN" # only for ODS>=12
|
||||||
|
}
|
||||||
|
|
||||||
|
def fill_dbo(con: Connection, map_dbo: Dict):
|
||||||
|
cur = con.cursor()
|
||||||
|
sql = """
|
||||||
|
select rel_id, rel_name, idx_id, idx_name
|
||||||
|
from (
|
||||||
|
select
|
||||||
|
rr.rdb$relation_id rel_id, -- 0
|
||||||
|
rr.rdb$relation_name rel_name, -- 1
|
||||||
|
-1 idx_id, -- 2
|
||||||
|
'' idx_name, -- 3
|
||||||
|
rr.rdb$relation_type rel_type,
|
||||||
|
rr.rdb$system_flag sys_flag
|
||||||
|
from rdb$relations rr
|
||||||
|
|
||||||
|
union all
|
||||||
|
|
||||||
|
select
|
||||||
|
rr.rdb$relation_id rel_id, -- 0
|
||||||
|
rr.rdb$relation_name rel_name, -- 1
|
||||||
|
coalesce(ri.rdb$index_id-1,-1) idx_id, -- 2
|
||||||
|
coalesce(ri.rdb$index_name,'') idx_name, -- 3
|
||||||
|
rr.rdb$relation_type rel_type,
|
||||||
|
rr.rdb$system_flag sys_flag
|
||||||
|
from rdb$relations rr
|
||||||
|
join rdb$indices ri on
|
||||||
|
rr.rdb$relation_name = ri.rdb$relation_name
|
||||||
|
) r
|
||||||
|
where
|
||||||
|
coalesce(r.rel_type,0) = 0 -- exclude views, GTT and external tables
|
||||||
|
and r.sys_flag is distinct from 1
|
||||||
|
"""
|
||||||
|
cur.execute(sql)
|
||||||
|
for r in cur:
|
||||||
|
map_dbo[r[0], r[2]] = (r[1].strip(), r[3].strip())
|
||||||
|
|
||||||
|
def parse_page_header(con: Connection, page_number: int, map_dbo: Dict):
|
||||||
|
page_buffer = con.info.get_page_content(page_number)
|
||||||
|
|
||||||
|
# dimitr, 20.01.2017 ~13:00
|
||||||
|
# all *CHAR = 1 byte, *SHORT = 2 bytes, *LONG = 4 bytes.
|
||||||
|
|
||||||
|
# https://docs.python.org/2/library/struct.html
|
||||||
|
# struct.unpack_from(fmt, buffer[, offset=0])
|
||||||
|
# Unpack the buffer according to the given format.
|
||||||
|
# The result is a tuple even if it contains exactly one item.
|
||||||
|
# The buffer must contain at least the amount of data required by the format
|
||||||
|
# len(buffer[offset:]) must be at least calcsize(fmt).
|
||||||
|
# First character of the format string can be used to indicate the byte order,
|
||||||
|
# size and alignment of the packed data
|
||||||
|
# Native byte order is big-endian or little-endian:
|
||||||
|
# < little-endian
|
||||||
|
# > big-endian
|
||||||
|
# Intel x86 and AMD64 (x86-64) are little-endian
|
||||||
|
# Use sys.byteorder to check the endianness of your system:
|
||||||
|
# https://docs.python.org/2/library/struct.html#format-characters
|
||||||
|
# c char string of length 1
|
||||||
|
# b signed char
|
||||||
|
# B unsigned char
|
||||||
|
# h short
|
||||||
|
# H unsigned short integer
|
||||||
|
# i int integer 4
|
||||||
|
# I unsigned int integer 4
|
||||||
|
# l long (4)
|
||||||
|
# L unsigned long (4)
|
||||||
|
# q long long (8)
|
||||||
|
# Q unsigned long long
|
||||||
|
|
||||||
|
page_type = unpack_from('<b', page_buffer)[0]
|
||||||
|
|
||||||
|
relation_id = -1
|
||||||
|
index_id = -1
|
||||||
|
segment_cnt = -1 # for Data page: number of record segments on page
|
||||||
|
index_id = -1
|
||||||
|
ix_level = -1
|
||||||
|
btr_len = -1
|
||||||
|
|
||||||
|
if page_type == 4:
|
||||||
|
# POINTER pege:
|
||||||
|
# *pag* dpg_header=16, SLONG dpg_sequence=4, SLONG ppg_next=4, USHORT ppg_count=2 ==> 16+4+4+2=26
|
||||||
|
# struct pointer_page
|
||||||
|
# {
|
||||||
|
# pag ppg_header;
|
||||||
|
# SLONG ppg_sequence; // Sequence number in relation
|
||||||
|
# SLONG ppg_next; // Next pointer page in relation
|
||||||
|
# USHORT ppg_count; // Number of slots active
|
||||||
|
# USHORT ppg_relation; // Relation id
|
||||||
|
# USHORT ppg_min_space; // Lowest slot with space available
|
||||||
|
# USHORT ppg_max_space; // Highest slot with space available
|
||||||
|
# SLONG ppg_page[1]; // Data page vector
|
||||||
|
# };
|
||||||
|
relation_id = unpack_from('<H', page_buffer, 26)[0] # 'H' ==> USHORT
|
||||||
|
elif page_type == 5:
|
||||||
|
# DATA page:
|
||||||
|
# *pag* dpg_header=16, SLONG dpg_sequence=4 ==> 16+4 = 20:
|
||||||
|
# struct data_page
|
||||||
|
# {
|
||||||
|
# 16 pag dpg_header;
|
||||||
|
# 4 SLONG dpg_sequence; // Sequence number in relation
|
||||||
|
# 2 USHORT dpg_relation; // Relation id
|
||||||
|
# 2 USHORT dpg_count; // Number of record segments on page
|
||||||
|
# struct dpg_repeat
|
||||||
|
# {
|
||||||
|
# USHORT dpg_offset; // Offset of record fragment
|
||||||
|
# USHORT dpg_length; // Length of record fragment
|
||||||
|
# } dpg_rpt[1];
|
||||||
|
# };
|
||||||
|
relation_id = unpack_from('<H', page_buffer, 20)[0] # 'H' ==> USHORT
|
||||||
|
segment_cnt = unpack_from('<H', page_buffer, 22)[0]
|
||||||
|
elif page_type == 6:
|
||||||
|
# Index root page
|
||||||
|
# struct index_root_page
|
||||||
|
# {
|
||||||
|
# pag irt_header;
|
||||||
|
# USHORT irt_relation; // relation id (for consistency)
|
||||||
|
relation_id = unpack_from('<H', page_buffer, 16)[0] # 'H' ==> USHORT
|
||||||
|
elif page_type == 7:
|
||||||
|
# B-tree page ("bucket"):
|
||||||
|
# struct btree_page
|
||||||
|
# {
|
||||||
|
# 16 pag btr_header;
|
||||||
|
# 4 SLONG btr_sibling; // right sibling page
|
||||||
|
# 4 SLONG btr_left_sibling; // left sibling page
|
||||||
|
# 4 SLONG btr_prefix_total; // sum of all prefixes on page
|
||||||
|
# 2 USHORT btr_relation; // relation id for consistency
|
||||||
|
# 2 USHORT btr_length; // length of data in bucket
|
||||||
|
# 1 UCHAR btr_id; // index id for consistency
|
||||||
|
# 1 UCHAR btr_level; // index level (0 = leaf)
|
||||||
|
# btree_nod btr_nodes[1];
|
||||||
|
# };
|
||||||
|
relation_id = unpack_from('<H', page_buffer, 28)[0] # 'H' ==> USHORT
|
||||||
|
btr_len = unpack_from('<H', page_buffer, 30)[0] # 'H' ==> USHORT // length of data in bucket
|
||||||
|
index_id = unpack_from('<B', page_buffer, 32)[0] # 'B' => UCHAR
|
||||||
|
ix_level = unpack_from('<B', page_buffer, 33)[0]
|
||||||
|
#
|
||||||
|
if index_id>=0 and (relation_id, index_id) in map_dbo:
|
||||||
|
u = map_dbo[ relation_id, index_id ]
|
||||||
|
page_info = f'{PAGE_TYPES[page_type].ljust(9)}, {u[1].strip()}, data_len={btr_len}, lev={ix_level}'
|
||||||
|
#page_info = ''.join((PAGE_TYPES[page_type].ljust(9), ', ', u[1].strip(), ', data_len=', str(btr_len), ', lev=', str(ix_level))) # 'Indx Page, <index_name>, <length of data in bucket>'
|
||||||
|
elif (relation_id, -1) in map_dbo:
|
||||||
|
u = map_dbo[ relation_id, -1 ]
|
||||||
|
if page_type == 5:
|
||||||
|
page_info = f'{PAGE_TYPES[page_type].ljust(9)}, {u[0].strip()}, segments on page: {segment_cnt}'
|
||||||
|
#page_info = ''.join( ( PAGE_TYPES[page_type].ljust(9),', ',u[0].strip(),', segments on page: ',str(segment_cnt) ) ) # '<table_name>, segments on page: NNN' - for Data page
|
||||||
|
else:
|
||||||
|
page_info = f'{PAGE_TYPES[page_type].ljust(9)}, {u[0].strip()}'
|
||||||
|
#page_info = ''.join( ( PAGE_TYPES[page_type].ljust(9),', ',u[0].strip() ) ) # '<table_name>' - for Pointer page
|
||||||
|
elif relation_id == -1:
|
||||||
|
page_info = PAGE_TYPES[page_type].ljust(9)
|
||||||
|
else:
|
||||||
|
page_info = f'UNKNOWN; {PAGE_TYPES[page_type].ljust(9)}; relation_id {relation_id}; index_id {index_id}'
|
||||||
|
#page_info = ''.join( ('UNKNOWN; ',PAGE_TYPES[page_type].ljust(9),'; relation_id ', str(relation_id), '; index_id ', str(index_id)) )
|
||||||
|
return (page_type, relation_id, page_info)
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.2')
|
@pytest.mark.version('>=3.0.2')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, capsys):
|
||||||
def test_1(db_1):
|
map_dbo = {}
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
sql = """
|
||||||
|
select p.rdb$relation_id, p.rdb$page_number
|
||||||
|
from rdb$pages p
|
||||||
|
join rdb$relations r on p.rdb$relation_id = r.rdb$relation_id
|
||||||
|
where r.rdb$relation_name=upper('TEST') and p.rdb$page_type = 4
|
||||||
|
order by p.rdb$page_number
|
||||||
|
rows 1
|
||||||
|
"""
|
||||||
|
with act_1.db.connect() as con:
|
||||||
|
fill_dbo(con, map_dbo)
|
||||||
|
c = con.cursor()
|
||||||
|
rel_id, pp1st = c.execute(sql).fetchone()
|
||||||
|
# Found first page for each of three types: Data, Index and Blob
|
||||||
|
# (loop starts from first PointerPage of table 'TEST')
|
||||||
|
brk_datapage = brk_indxpage = brk_blobpage = -1
|
||||||
|
for i in range(pp1st, con.info.pages_allocated):
|
||||||
|
page_type, relation_id, page_info = parse_page_header(con, i, map_dbo)
|
||||||
|
#print('page:',i, '; page_type:',page_type, '; rel_id:',relation_id,';', page_info)
|
||||||
|
if relation_id == 128 and page_type == 5:
|
||||||
|
brk_datapage = i
|
||||||
|
elif relation_id == 128 and page_type == 7:
|
||||||
|
brk_indxpage = i
|
||||||
|
elif page_type == 8:
|
||||||
|
brk_blobpage = i
|
||||||
|
if brk_datapage > 0 and brk_indxpage > 0 and brk_blobpage > 0:
|
||||||
|
break
|
||||||
|
#
|
||||||
|
# Store binary content of .fdb for futher restore
|
||||||
|
raw_db_content = act_1.db.db_path.read_bytes()
|
||||||
|
# Make pages damaged
|
||||||
|
# 0xFFAACCEEBB0000CC 0xDDEEAADDCC00DDEE
|
||||||
|
bw = bytearray(b'\\xff\\xaa\\xcc\\xee\\xbb\\x00\\x00\\xcc\\xdd\\xee\\xaa\\xdd\\xcc\\x00\\xdd\\xee')
|
||||||
|
with open(act_1.db.db_path, 'r+b') as w:
|
||||||
|
for brk_page in (brk_datapage, brk_indxpage, brk_blobpage):
|
||||||
|
w.seek(brk_page * con.info.page_size)
|
||||||
|
w.write(bw)
|
||||||
|
#
|
||||||
|
act_1.gstat(switches=['-e'])
|
||||||
|
pattern = re.compile('(data|index|blob|other)\\s+pages[:]{0,1}\\s+total[:]{0,1}\\s+\\d+[,]{0,1}\\s+encrypted[:]{0,1}\\s+\\d+.*[,]{0,1}non-crypted[:]{0,1}\\s+\\d+.*', re.IGNORECASE)
|
||||||
|
for line in act_1.stdout.splitlines():
|
||||||
|
if pattern.match(line.strip()):
|
||||||
|
print(line.strip())
|
||||||
|
# Validate DB - ensure that there are errors in pages
|
||||||
|
# RESULT: validation log should contain lines with problems about three diff. page types:
|
||||||
|
# expected data encountered unknown
|
||||||
|
# expected index B-tree encountered unknown
|
||||||
|
# expected blob encountered unknown
|
||||||
|
with act_1.connect_server() as srv:
|
||||||
|
srv.database.validate(database=act_1.db.db_path, lock_timeout=1)
|
||||||
|
validation_log = srv.readlines()
|
||||||
|
# Process validation log
|
||||||
|
data_page_problem = indx_page_problem = blob_page_problem = False
|
||||||
|
for line in validation_log:
|
||||||
|
if 'expected data' in line:
|
||||||
|
data_page_problem = True
|
||||||
|
elif 'expected index B-tree' in line:
|
||||||
|
indx_page_problem = True
|
||||||
|
elif 'expected blob' in line:
|
||||||
|
blob_page_problem = True
|
||||||
|
print(f"Detected all THREE page types with problem => {'YES' if data_page_problem and indx_page_problem and blob_page_problem else 'NO'}")
|
||||||
|
# restore DB content
|
||||||
|
act_1.db.db_path.write_bytes(raw_db_content)
|
||||||
|
# Check
|
||||||
|
act_1.reset()
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.stdout = capsys.readouterr().out
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -22,12 +22,14 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
|
||||||
# version: 4.0
|
# version: 4.0
|
||||||
# resources: None
|
# resources: None
|
||||||
|
|
||||||
substitutions_1 = [('[ \t]+', ' ')]
|
substitutions_1 = [('[ \t]+', ' ')]
|
||||||
|
#substitutions_1 = []
|
||||||
|
|
||||||
init_script_1 = """
|
init_script_1 = """
|
||||||
recreate view v_test as select 1 x from rdb$database;
|
recreate view v_test as select 1 x from rdb$database;
|
||||||
@ -76,7 +78,6 @@ init_script_1 = """
|
|||||||
;
|
;
|
||||||
commit;
|
commit;
|
||||||
|
|
||||||
|
|
||||||
insert into test_anna default values;
|
insert into test_anna default values;
|
||||||
insert into test_beta default values;
|
insert into test_beta default values;
|
||||||
insert into test_ciao default values;
|
insert into test_ciao default values;
|
||||||
@ -95,7 +96,6 @@ init_script_1 = """
|
|||||||
insert into test_won2 default values;
|
insert into test_won2 default values;
|
||||||
insert into test_w_n3 default values;
|
insert into test_w_n3 default values;
|
||||||
commit;
|
commit;
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
||||||
@ -110,8 +110,7 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
# os.environ["ISC_USER"] = user_name
|
# os.environ["ISC_USER"] = user_name
|
||||||
# os.environ["ISC_PASSWORD"] = user_password
|
# os.environ["ISC_PASSWORD"] = user_password
|
||||||
#
|
#
|
||||||
# # dsn localhost/3400:C:\\FBTESTING\\qa
|
# # dsn localhost/3400:C:\\FBTESTING\\qa\\fbt-repo\\tmp\\bugs.core_NNNN.fdb
|
||||||
# bt-repo mpugs.core_NNNN.fdb
|
|
||||||
# # db_conn.database_name C:\\FBTESTING\\QA\\FBT-REPO\\TMP\\BUGS.CORE_NNNN.FDB
|
# # db_conn.database_name C:\\FBTESTING\\QA\\FBT-REPO\\TMP\\BUGS.CORE_NNNN.FDB
|
||||||
# # $(DATABASE_LOCATION)... C:/FBTESTING/qa/fbt-repo/tmp/bugs.core_NNN.fdb
|
# # $(DATABASE_LOCATION)... C:/FBTESTING/qa/fbt-repo/tmp/bugs.core_NNN.fdb
|
||||||
#
|
#
|
||||||
@ -163,7 +162,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
0 test_doc% doca
|
0 test_doc% doca
|
||||||
@ -181,9 +181,47 @@ expected_stdout_1 = """
|
|||||||
1 test_(a|b)[[:ALPHA:]]+a test_d(o|u)% beta
|
1 test_(a|b)[[:ALPHA:]]+a test_d(o|u)% beta
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
# this_fbk=os.path.join(context['temp_directory'],'tmp_5538.fbk')
|
||||||
|
# test_res=os.path.join(context['temp_directory'],'tmp_5538.tmp')
|
||||||
|
|
||||||
|
fbk_file_1 = temp_file('core_5538.fbk')
|
||||||
|
fdb_file_1 = temp_file('core_5538.fdb')
|
||||||
|
|
||||||
@pytest.mark.version('>=4.0')
|
@pytest.mark.version('>=4.0')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, fbk_file_1: Path, fdb_file_1: Path, capsys):
|
||||||
def test_1(db_1):
|
# 1. Check that we can use patterns for include data only from several selected tables:
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
for i, p in enumerate(['test_doc%', 'test_d(o|u)ra', '%_w(i|o|_)n[[:DIGIT:]]', 'test_a[[:ALPHA:]]{1,}a']):
|
||||||
|
act_1.reset()
|
||||||
|
act_1.gbak(switches=['-b', act_1.db.dsn, str(fbk_file_1), '-include', p])
|
||||||
|
act_1.reset()
|
||||||
|
act_1.gbak(switches=['-rep', str(fbk_file_1), f'localhost:{fdb_file_1}'])
|
||||||
|
act_1.reset()
|
||||||
|
act_1.isql(switches=[f'localhost:{fdb_file_1}'], connect_db=False,
|
||||||
|
input=f"set heading off; select {i} ptn_indx, q'{{{p}}}' as ptn_text, v.* from v_test v;")
|
||||||
|
print(act_1.stdout)
|
||||||
|
# 2. Check interaction between -INCLUDE_DATA and -SKIP_DATA switches for a table:
|
||||||
|
# We must check only conditions marked by '**':
|
||||||
|
# +--------------------------------------------------+
|
||||||
|
# | | INCLUDE_DATA |
|
||||||
|
# | |--------------------------------------|
|
||||||
|
# | SKIP_DATA | NOT SET | MATCH | NOT MATCH |
|
||||||
|
# +-----------+------------+------------+------------+
|
||||||
|
# | NOT SET | included | included | excluded | <<< these rules can be skipped in this test
|
||||||
|
# | MATCH | excluded |**excluded**|**excluded**|
|
||||||
|
# | NOT MATCH | included |**included**|**excluded**|
|
||||||
|
# +-----------+------------+------------+------------+
|
||||||
|
skip_ptn = 'test_d(o|u)%'
|
||||||
|
for i, p in enumerate(['test_d%', 'test_(a|b)[[:ALPHA:]]+a']):
|
||||||
|
act_1.reset()
|
||||||
|
act_1.gbak(switches=['-b', act_1.db.dsn, str(fbk_file_1), '-include_data', p, '-skip_data', skip_ptn])
|
||||||
|
act_1.reset()
|
||||||
|
act_1.gbak(switches=['-rep', str(fbk_file_1), f'localhost:{fdb_file_1}'])
|
||||||
|
act_1.reset()
|
||||||
|
act_1.isql(switches=[f'localhost:{fdb_file_1}'], connect_db=False,
|
||||||
|
input=f"set heading off; select {i} ptn_indx, q'{{{p}}}' as include_ptn, q'{{{skip_ptn}}}' as exclude_ptn, v.* from v_test v;")
|
||||||
|
print(act_1.stdout)
|
||||||
|
# Check
|
||||||
|
act_1.reset()
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.stdout = capsys.readouterr().out
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -20,7 +20,8 @@
|
|||||||
# qmid:
|
# qmid:
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from decimal import Decimal
|
||||||
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 3.0.3
|
# version: 3.0.3
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -62,7 +63,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
X -Infinity
|
X -Infinity
|
||||||
@ -71,8 +73,11 @@ expected_stdout_1 = """
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.3')
|
@pytest.mark.version('>=3.0.3')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action):
|
||||||
def test_1(db_1):
|
with act_1.db.connect() as con:
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
c = con.cursor()
|
||||||
|
c.execute('insert into test(x, y) values(?, ?)', [Decimal('-Infinity'), Decimal('Infinity')])
|
||||||
|
con.commit()
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.isql(switches=[], input="set list on; set count on; select * from test;")
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -30,12 +30,14 @@
|
|||||||
# qmid:
|
# qmid:
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
|
||||||
# version: 3.0.3
|
# version: 3.0.3
|
||||||
# resources: None
|
# resources: None
|
||||||
|
|
||||||
substitutions_1 = [('[0-9][0-9]:[0-9][0-9]:[0-9][0-9].[0-9][0-9]', ''), ('Relation [0-9]{3,4}', 'Relation')]
|
substitutions_1 = [('[0-9][0-9]:[0-9][0-9]:[0-9][0-9].[0-9][0-9]', ''),
|
||||||
|
('Relation [0-9]{3,4}', 'Relation')]
|
||||||
|
|
||||||
init_script_1 = """
|
init_script_1 = """
|
||||||
recreate table test (
|
recreate table test (
|
||||||
@ -157,11 +159,14 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
# cleanup( (f_bkrs_err, f_val_log, f_val_err, f_bkup_tmp, f_rest_tmp) )
|
# cleanup( (f_bkrs_err, f_val_log, f_val_err, f_bkup_tmp, f_rest_tmp) )
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
|
||||||
expected_stdout_1 = """
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
X1 1
|
|
||||||
|
expected_stdout_1_a = """
|
||||||
X1 1
|
X1 1
|
||||||
|
"""
|
||||||
|
|
||||||
|
expected_stdout_1_b = """
|
||||||
Validation started
|
Validation started
|
||||||
Relation 128 (TEST)
|
Relation 128 (TEST)
|
||||||
process pointer page 0 of 1
|
process pointer page 0 of 1
|
||||||
@ -170,9 +175,27 @@ expected_stdout_1 = """
|
|||||||
Validation finished
|
Validation finished
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
fbk_file_1 = temp_file('core_5576.fbk')
|
||||||
|
fdb_file_1 = temp_file('core_5576.fdb')
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.3')
|
@pytest.mark.version('>=3.0.3')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, fbk_file_1: Path, fdb_file_1: Path):
|
||||||
def test_1(db_1):
|
act_1.gbak(switches=['-b', act_1.db.dsn, str(fbk_file_1)])
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
act_1.reset()
|
||||||
|
act_1.gbak(switches=['-rep', str(fbk_file_1), f'localhost:{fdb_file_1}'])
|
||||||
|
#
|
||||||
|
for i in range(2): # Run isql twice!
|
||||||
|
act_1.reset()
|
||||||
|
act_1.expected_stdout = expected_stdout_1_a
|
||||||
|
act_1.isql(switches=[f'localhost:{fdb_file_1}'], connect_db=False,
|
||||||
|
input='set list on;select 1 x1 from test where i=1 with lock;')
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
# Validate the database
|
||||||
|
act_1.reset()
|
||||||
|
act_1.expected_stdout = expected_stdout_1_b
|
||||||
|
with act_1.connect_server() as srv:
|
||||||
|
srv.database.validate(database=fdb_file_1)
|
||||||
|
act_1.stdout = ''.join(srv.readlines())
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
|
||||||
|
|
||||||
|
@ -38,7 +38,11 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
import re
|
||||||
|
import zipfile
|
||||||
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
from firebird.driver import SrvRestoreFlag
|
||||||
|
|
||||||
# version: 2.5.8
|
# version: 2.5.8
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -155,12 +159,23 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
|
fbk_file_1 = temp_file('core_5579_broken_nn.fbk')
|
||||||
|
fdb_file_1 = temp_file('core_5579_broken_nn.fdb')
|
||||||
|
|
||||||
@pytest.mark.version('>=2.5.8')
|
@pytest.mark.version('>=2.5.8')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, fdb_file_1: Path, fbk_file_1: Path):
|
||||||
def test_1(db_1):
|
pattern = re.compile('[.*]*request\\s+synchronization\\s+error\\.*', re.IGNORECASE)
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
zipped_fbk_file = zipfile.Path(act_1.vars['files'] / 'core_5579_broken_nn.zip',
|
||||||
|
at='core_5579_broken_nn.fbk')
|
||||||
|
fbk_file_1.write_bytes(zipped_fbk_file.read_bytes())
|
||||||
|
with act_1.connect_server() as srv:
|
||||||
|
srv.database.restore(database=fdb_file_1, backup=fbk_file_1,
|
||||||
|
flags=SrvRestoreFlag.ONE_AT_A_TIME | SrvRestoreFlag.CREATE)
|
||||||
|
|
||||||
|
# before this ticket was fixed restore fails with: request synchronization error
|
||||||
|
for line in srv:
|
||||||
|
if pattern.search(line):
|
||||||
|
pytest.fail(f'RESTORE ERROR: {line}')
|
||||||
|
@ -26,12 +26,13 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 3.0.3
|
# version: 3.0.3
|
||||||
# resources: None
|
# resources: None
|
||||||
|
|
||||||
substitutions_1 = [('[ \t]+', ' '), ('.*RECORD LENGTH:[ \t]+[\\d]+[ \t]*\\)', ''), ('.*COUNT[ \t]+[\\d]+', '')]
|
substitutions_1 = [('[ \t]+', ' '), ('.*RECORD LENGTH:[ \t]+[\\d]+[ \t]*\\)', ''),
|
||||||
|
('.*COUNT[ \t]+[\\d]+', '')]
|
||||||
|
|
||||||
init_script_1 = """"""
|
init_script_1 = """"""
|
||||||
|
|
||||||
@ -163,22 +164,50 @@ db_1 = db_factory(charset='UTF8', sql_dialect=3, init=init_script_1)
|
|||||||
# cleanup( ( isql_run, isql_log, isql_err ) )
|
# cleanup( ( isql_run, isql_log, isql_err ) )
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
STDOUT:SELECT EXPRESSION
|
SELECT EXPRESSION
|
||||||
STDOUT: -> AGGREGATE
|
-> AGGREGATE
|
||||||
STDOUT: -> FILTER
|
-> FILTER
|
||||||
STDOUT: -> HASH JOIN (INNER)
|
-> HASH JOIN (INNER)
|
||||||
STDOUT: -> TABLE "TEST" AS "B" FULL SCAN
|
-> TABLE "TEST" AS "B" FULL SCAN
|
||||||
STDOUT: -> RECORD BUFFER (RECORD LENGTH: 32793)
|
-> RECORD BUFFER (RECORD LENGTH: 32793)
|
||||||
STDOUT: -> TABLE "TEST" AS "A" FULL SCAN
|
-> TABLE "TEST" AS "A" FULL SCAN
|
||||||
STDOUT:COUNT 17000
|
COUNT 17000
|
||||||
|
"""
|
||||||
|
|
||||||
|
MIN_RECS_TO_ADD = 17000
|
||||||
|
|
||||||
|
test_script_1 = """
|
||||||
|
set list on;
|
||||||
|
--show version;
|
||||||
|
set explain on;
|
||||||
|
select count(*) from test a join test b using(id, s);
|
||||||
|
set explain off;
|
||||||
|
quit;
|
||||||
|
select
|
||||||
|
m.MON$STAT_ID
|
||||||
|
,m.MON$STAT_GROUP
|
||||||
|
,m.MON$MEMORY_USED
|
||||||
|
,m.MON$MEMORY_ALLOCATED
|
||||||
|
,m.MON$MAX_MEMORY_USED
|
||||||
|
,m.MON$MAX_MEMORY_ALLOCATED
|
||||||
|
from mon$database d join mon$memory_usage m using (MON$STAT_ID);
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.3')
|
@pytest.mark.version('>=3.0.3')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action):
|
||||||
def test_1(db_1):
|
act_1.db.set_async_write()
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
with act_1.db.connect(charset='utf8') as con:
|
||||||
|
con.execute_immediate('create table test(id int, s varchar(8191))')
|
||||||
|
con.commit()
|
||||||
|
c = con.cursor()
|
||||||
|
c.execute(f"insert into test(id, s) select row_number()over(), lpad('', 8191, 'Алексей, Łukasz, Máté, François, Jørgen, Νικόλαος') from rdb$types,rdb$types rows {MIN_RECS_TO_ADD}")
|
||||||
|
con.commit()
|
||||||
|
#
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.isql(switches=[], input=test_script_1)
|
||||||
|
act_1.stdout = act_1.stdout.upper()
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -23,7 +23,11 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
import zipfile
|
||||||
|
from difflib import unified_diff
|
||||||
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
from firebird.driver import SrvRepairFlag
|
||||||
|
|
||||||
# version: 3.0.3
|
# version: 3.0.3
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -199,15 +203,38 @@ db_1 = db_factory(page_size=4096, sql_dialect=3, init=init_script_1)
|
|||||||
# cleanup( (f_fblog_before, f_fblog_after, f_diff_txt, val_log, blob_handle, fwoff_log) )
|
# cleanup( (f_fblog_before, f_fblog_after, f_diff_txt, val_log, blob_handle, fwoff_log) )
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
DIFF IN FIREBIRD.LOG: + VALIDATION FINISHED: 0 ERRORS, 0 WARNINGS, 0 FIXED
|
+ VALIDATION FINISHED: 0 ERRORS, 0 WARNINGS, 0 FIXED
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
blob_src_1 = temp_file('core_5618.bin')
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.3')
|
@pytest.mark.version('>=3.0.3')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, blob_src_1: Path):
|
||||||
def test_1(db_1):
|
act_1.db.set_async_write()
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
zipped_blob_file = zipfile.Path(act_1.vars['files'] / 'core_5618.zip',
|
||||||
|
at='core_5618.bin')
|
||||||
|
blob_src_1.write_bytes(zipped_blob_file.read_bytes())
|
||||||
|
#
|
||||||
|
with act_1.db.connect() as con:
|
||||||
|
c = con.cursor()
|
||||||
|
with open(blob_src_1, mode='rb') as blob_handle:
|
||||||
|
c.execute('insert into test (b) values (?)', [blob_handle])
|
||||||
|
c.close()
|
||||||
|
con.execute_immediate('drop table test')
|
||||||
|
con.commit()
|
||||||
|
#
|
||||||
|
log_before = act_1.get_firebird_log()
|
||||||
|
# Run full validation (this is what 'gfix -v -full' does)
|
||||||
|
with act_1.connect_server() as srv:
|
||||||
|
srv.database.repair(database=act_1.db.db_path,
|
||||||
|
flags=SrvRepairFlag.FULL | SrvRepairFlag.VALIDATE_DB)
|
||||||
|
assert srv.readlines() == []
|
||||||
|
#
|
||||||
|
log_after = act_1.get_firebird_log()
|
||||||
|
log_diff = [line.strip().upper() for line in unified_diff(log_before, log_after)
|
||||||
|
if line.startswith('+') and 'WARNING' in line.upper()]
|
||||||
|
assert log_diff == ['+\tVALIDATION FINISHED: 0 ERRORS, 0 WARNINGS, 0 FIXED']
|
||||||
|
@ -23,7 +23,8 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
|
||||||
# version: 3.0.3
|
# version: 3.0.3
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -121,16 +122,19 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
|
||||||
expected_stdout_1 = """
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
|
expected_stdout_1_a = """
|
||||||
RDB$FILE_SEQUENCE 0
|
RDB$FILE_SEQUENCE 0
|
||||||
RDB$FILE_START 0
|
RDB$FILE_START 0
|
||||||
RDB$FILE_LENGTH 0
|
RDB$FILE_LENGTH 0
|
||||||
RDB$FILE_FLAGS 1
|
RDB$FILE_FLAGS 1
|
||||||
RDB$SHADOW_NUMBER 1
|
RDB$SHADOW_NUMBER 1
|
||||||
S_HASH_BEFORE 1499836372373901520
|
S_HASH_BEFORE 1499836372373901520
|
||||||
|
"""
|
||||||
|
|
||||||
|
expected_stdout_1_b = """
|
||||||
RDB$FILE_SEQUENCE 0
|
RDB$FILE_SEQUENCE 0
|
||||||
RDB$FILE_START 0
|
RDB$FILE_START 0
|
||||||
RDB$FILE_LENGTH 0
|
RDB$FILE_LENGTH 0
|
||||||
@ -139,9 +143,62 @@ expected_stdout_1 = """
|
|||||||
S_HASH_AFTER 1499836372373901520
|
S_HASH_AFTER 1499836372373901520
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
fdb_file_1 = temp_file('core_5630.fdb')
|
||||||
|
fbk_file_1 = temp_file('core_5630.fbk')
|
||||||
|
shd_file_1 = temp_file('core_5630.shd')
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.3')
|
@pytest.mark.version('>=3.0.3')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, fdb_file_1: Path, fbk_file_1: Path, shd_file_1: Path):
|
||||||
def test_1(db_1):
|
init_ddl = f"""
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
set bail on;
|
||||||
|
set list on;
|
||||||
|
|
||||||
|
create database 'localhost:{fdb_file_1}' user '{act_1.db.user}' password '{act_1.db.password}';
|
||||||
|
|
||||||
|
recreate table test(s varchar(30));
|
||||||
|
commit;
|
||||||
|
|
||||||
|
create or alter view v_shadow_info as
|
||||||
|
select
|
||||||
|
rdb$file_sequence -- 0
|
||||||
|
,rdb$file_start -- 0
|
||||||
|
,rdb$file_length -- 0
|
||||||
|
,rdb$file_flags -- 1
|
||||||
|
,rdb$shadow_number -- 1
|
||||||
|
from rdb$files
|
||||||
|
where lower(rdb$file_name) containing lower('core_5630.shd')
|
||||||
|
;
|
||||||
|
|
||||||
|
insert into test select 'line #' || lpad(row_number()over(), 3, '0' ) from rdb$types rows 200;
|
||||||
|
commit;
|
||||||
|
|
||||||
|
create shadow 1 '{shd_file_1}';
|
||||||
|
commit;
|
||||||
|
set list on;
|
||||||
|
select * from v_shadow_info;
|
||||||
|
select hash( list(s) ) as s_hash_before from test;
|
||||||
|
quit;
|
||||||
|
"""
|
||||||
|
act_1.expected_stdout = expected_stdout_1_a
|
||||||
|
act_1.isql(switches=['-q'], input=init_ddl)
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
#
|
||||||
|
with act_1.connect_server() as srv:
|
||||||
|
srv.database.backup(database=fdb_file_1, backup=fbk_file_1)
|
||||||
|
srv.wait()
|
||||||
|
#
|
||||||
|
fdb_file_1.unlink()
|
||||||
|
shd_file_1.unlink()
|
||||||
|
#
|
||||||
|
act_1.reset()
|
||||||
|
act_1.gbak(switches=['-c', '-use_all_space', str(fbk_file_1), f'localhost:{fdb_file_1}'])
|
||||||
|
# Check that we have the same data in DB tables
|
||||||
|
sql_text = """
|
||||||
|
set list on;
|
||||||
|
select * from v_shadow_info;
|
||||||
|
select hash( list(s) ) as s_hash_after from test;
|
||||||
|
"""
|
||||||
|
act_1.reset()
|
||||||
|
act_1.expected_stdout = expected_stdout_1_b
|
||||||
|
act_1.isql(switches=['-q', f'localhost:{fdb_file_1}'], input=sql_text, connect_db=False)
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -22,7 +22,11 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
import zipfile
|
||||||
|
from difflib import unified_diff
|
||||||
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
from firebird.driver import SrvRestoreFlag
|
||||||
|
|
||||||
# version: 4.0
|
# version: 4.0
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -207,12 +211,29 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
|
sec_fbk_1 = temp_file('core5637-security3.fbk')
|
||||||
|
sec_fdb_1 = temp_file('core5637-security3.fdb')
|
||||||
|
|
||||||
@pytest.mark.version('>=4.0')
|
@pytest.mark.version('>=4.0')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, sec_fbk_1: Path, sec_fdb_1: Path):
|
||||||
def test_1(db_1):
|
zipped_fbk_file = zipfile.Path(act_1.vars['files'] / 'core_5637.zip',
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
at='core5637-security3.fbk')
|
||||||
|
sec_fbk_1.write_bytes(zipped_fbk_file.read_bytes())
|
||||||
|
#
|
||||||
|
log_before = act_1.get_firebird_log()
|
||||||
|
# Restore security database
|
||||||
|
with act_1.connect_server() as srv:
|
||||||
|
srv.database.restore(database=sec_fdb_1, backup=sec_fbk_1, flags=SrvRestoreFlag.REPLACE)
|
||||||
|
restore_log = srv.readlines()
|
||||||
|
#
|
||||||
|
log_after = act_1.get_firebird_log()
|
||||||
|
#
|
||||||
|
srv.database.validate(database=sec_fdb_1)
|
||||||
|
validation_log = srv.readlines()
|
||||||
|
#
|
||||||
|
assert [line for line in restore_log if 'ERROR' in line.upper()] == []
|
||||||
|
assert [line for line in validation_log if 'ERROR' in line.upper()] == []
|
||||||
|
assert list(unified_diff(log_before, log_after)) == []
|
||||||
|
@ -17,14 +17,21 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action, Database
|
||||||
|
|
||||||
# version: 3.0.3
|
# version: 3.0.3
|
||||||
# resources: None
|
# resources: None
|
||||||
|
|
||||||
substitutions_1 = [('INFO_BLOB_ID.*', '')]
|
substitutions_1 = [('INFO_BLOB_ID.*', '')]
|
||||||
|
|
||||||
init_script_1 = """"""
|
init_script_1 = """
|
||||||
|
create table persons (
|
||||||
|
id integer not null,
|
||||||
|
name varchar(60) not null,
|
||||||
|
address varchar(60),
|
||||||
|
info blob sub_type text
|
||||||
|
);
|
||||||
|
"""
|
||||||
|
|
||||||
db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
||||||
|
|
||||||
@ -142,7 +149,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
ID 1
|
ID 1
|
||||||
@ -159,9 +167,47 @@ expected_stdout_1 = """
|
|||||||
Records affected: 2
|
Records affected: 2
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
db_1_repl = db_factory(sql_dialect=3, init=init_script_1, filename='tmp_5645_repl.fd')
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.3')
|
@pytest.mark.version('>=3.0.3')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, db_1_repl: Database):
|
||||||
def test_1(db_1):
|
pytest.skip("Requires UDR udrcpp_example")
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
ddl_for_replication = f"""
|
||||||
|
create table replicate_config (
|
||||||
|
name varchar(31) not null,
|
||||||
|
data_source varchar(255) not null
|
||||||
|
);
|
||||||
|
|
||||||
|
insert into replicate_config (name, data_source)
|
||||||
|
values ('ds1', '{db_1_repl}');
|
||||||
|
|
||||||
|
create trigger persons_replicate
|
||||||
|
after insert on persons
|
||||||
|
external name 'udrcpp_example!replicate!ds1'
|
||||||
|
engine udr;
|
||||||
|
|
||||||
|
create trigger persons_replicate2
|
||||||
|
after insert on persons
|
||||||
|
external name 'udrcpp_example!replicate_persons!ds1'
|
||||||
|
engine udr;
|
||||||
|
commit;
|
||||||
|
"""
|
||||||
|
act_1.isql(switches=['-q'], input=ddl_for_replication)
|
||||||
|
#
|
||||||
|
with act_1.db.connect() as con:
|
||||||
|
c = con.cursor()
|
||||||
|
c.execute("insert into persons values (1, 'One', 'some_address', 'some_blob_info')")
|
||||||
|
con.commit()
|
||||||
|
#
|
||||||
|
if act_1.is_version('>4.0'):
|
||||||
|
act_1.reset()
|
||||||
|
act_1.isql(switches=['-q'], input='ALTER EXTERNAL CONNECTIONS POOL CLEAR ALL;')
|
||||||
|
# Check
|
||||||
|
act_1.reset()
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.isql(switches=['-q', db_1_repl.dsn],
|
||||||
|
input='set list on; set count on; select id,name,address,info as info_blob_id from persons; rollback;',
|
||||||
|
connect_db=False)
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
# qmid:
|
# qmid:
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 4.0
|
# version: 4.0
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -71,15 +71,48 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
# runProgram('isql',[dsn],script)
|
# runProgram('isql',[dsn],script)
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
RET_CODE 0
|
RET_CODE 0
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
test_script_1 = """
|
||||||
|
set bail on;
|
||||||
|
set list on;
|
||||||
|
set term ^;
|
||||||
|
execute block returns(ret_code smallint) as
|
||||||
|
declare n int = 300;
|
||||||
|
begin
|
||||||
|
while (n > 0) do
|
||||||
|
begin
|
||||||
|
if (mod(n, 2) = 0) then
|
||||||
|
begin
|
||||||
|
in autonomous transaction do
|
||||||
|
begin
|
||||||
|
execute statement 'create or alter view vw1 (dump1) as select 1 from rdb$database';
|
||||||
|
end
|
||||||
|
end
|
||||||
|
else
|
||||||
|
begin
|
||||||
|
in autonomous transaction do
|
||||||
|
begin
|
||||||
|
execute statement 'create or alter view vw1 (dump1, dump2) as select 1, 2 from rdb$database';
|
||||||
|
end
|
||||||
|
end
|
||||||
|
n = n - 1;
|
||||||
|
end
|
||||||
|
ret_code = -abs(n);
|
||||||
|
suspend;
|
||||||
|
end ^
|
||||||
|
set term ;^
|
||||||
|
quit;
|
||||||
|
"""
|
||||||
|
|
||||||
@pytest.mark.version('>=4.0')
|
@pytest.mark.version('>=4.0')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action):
|
||||||
def test_1(db_1):
|
act_1.db.set_async_write()
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.isql(switches=[], input=test_script_1)
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -35,7 +35,11 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
import subprocess
|
||||||
|
import time
|
||||||
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
from firebird.driver import ShutdownMode, ShutdownMethod
|
||||||
|
|
||||||
# version: 3.0.3
|
# version: 3.0.3
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -238,18 +242,68 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
STDLOG of 2nd EDS: RESULT_MSG OK: second EDS was fast
|
RESULT_MSG OK: second EDS was fast
|
||||||
STDLOG of DB reset: Point before DB shutdown.
|
|
||||||
STDLOG of DB reset: Point after DB shutdown.
|
|
||||||
STDLOG of DB reset: Point after DB online.
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
eds_script_1 = temp_file('eds_script.sql')
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.3')
|
@pytest.mark.version('>=3.0.3')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, eds_script_1: Path):
|
||||||
def test_1(db_1):
|
eds_sql = f"""
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
set bail on;
|
||||||
|
set list on;
|
||||||
|
--select current_timestamp as " " from rdb$database;
|
||||||
|
set term ^;
|
||||||
|
execute block as
|
||||||
|
declare c smallint;
|
||||||
|
declare remote_host varchar(50) = '%(remote_host)s'; -- never unreachable: 192.0.2.2
|
||||||
|
begin
|
||||||
|
rdb$set_context('USER_SESSION','DTS_BEG', cast('now' as timestamp) );
|
||||||
|
execute statement 'select 1 from rdb$database'
|
||||||
|
on external remote_host || ':' || rdb$get_context('SYSTEM', 'DB_NAME')
|
||||||
|
as user '{act_1.db.user}' password '{act_1.db.password}'
|
||||||
|
into c;
|
||||||
|
end
|
||||||
|
^
|
||||||
|
set term ;^
|
||||||
|
--select current_timestamp as " " from rdb$database;
|
||||||
|
select iif( waited_ms < max_wait_ms,
|
||||||
|
'OK: second EDS was fast',
|
||||||
|
'FAILED: second EDS waited too long, ' || waited_ms || ' ms - more than max_wait_ms='||max_wait_ms
|
||||||
|
) as result_msg
|
||||||
|
from (
|
||||||
|
select
|
||||||
|
datediff( millisecond from cast( rdb$get_context('USER_SESSION','DTS_BEG') as timestamp) to current_timestamp ) as waited_ms
|
||||||
|
,500 as max_wait_ms
|
||||||
|
-- ^
|
||||||
|
-- | #################
|
||||||
|
-- +-------------------------------- T H R E S H O L D
|
||||||
|
-- #################
|
||||||
|
from rdb$database
|
||||||
|
);
|
||||||
|
"""
|
||||||
|
#
|
||||||
|
remote_host = '192.0.2.2'
|
||||||
|
eds_script_1.write_text(eds_sql % locals())
|
||||||
|
p_unavail_host = subprocess.Popen([act_1.vars['isql'], '-n', '-i', str(eds_script_1),
|
||||||
|
'-user', act_1.db.user,
|
||||||
|
'-password', act_1.db.password, act_1.db.dsn],
|
||||||
|
stdout=subprocess.DEVNULL, stderr=subprocess.STDOUT)
|
||||||
|
try:
|
||||||
|
time.sleep(2)
|
||||||
|
remote_host = 'localhost'
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.isql(switches=['-n'], input=eds_sql % locals())
|
||||||
|
finally:
|
||||||
|
p_unavail_host.terminate()
|
||||||
|
# Ensure that database is not busy
|
||||||
|
with act_1.connect_server() as srv:
|
||||||
|
srv.database.shutdown(database=act_1.db.db_path, mode=ShutdownMode.FULL,
|
||||||
|
method=ShutdownMethod.FORCED, timeout=0)
|
||||||
|
srv.database.bring_online(database=act_1.db.db_path)
|
||||||
|
# Check
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -20,7 +20,9 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
import zipfile
|
||||||
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
|
||||||
# version: 3.0
|
# version: 3.0
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -129,15 +131,40 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
# cleanup( (f_restore_log, tmpfdb, tmpfbk) )
|
# cleanup( (f_restore_log, tmpfdb, tmpfbk) )
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
PLAN JOIN (B INDEX (COM_PEDIDO_IDX1), A INDEX (FK_COM_PEDIDO_ITEM_PEDIDO), C INDEX (PK_EST_PRODUTO))
|
PLAN JOIN (B INDEX (COM_PEDIDO_IDX1), A INDEX (FK_COM_PEDIDO_ITEM_PEDIDO), C INDEX (PK_EST_PRODUTO))
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
test_script_1 = """
|
||||||
|
set planonly;
|
||||||
|
select
|
||||||
|
a.id_pedido_item,
|
||||||
|
c.descricao
|
||||||
|
from com_pedido b
|
||||||
|
join com_pedido_item a on a.id_pedido = b.id_pedido
|
||||||
|
and ( not(a.id_produto =1 and a.id_pedido_item_pai is not null))
|
||||||
|
join est_produto c on c.id_produto = a.id_produto
|
||||||
|
where
|
||||||
|
-- b.dth_pedido between cast('10.12.16 05:00:00' as timestamp) and cast('10.12.16 20:00:00' as timestamp)
|
||||||
|
b.dth_pedido between ? and ? ;
|
||||||
|
"""
|
||||||
|
|
||||||
|
fbk_file_1 = temp_file('core5637-security3.fbk')
|
||||||
|
fdb_file_1 = temp_file('bad_plan_5659.fdb')
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0')
|
@pytest.mark.version('>=3.0')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, fbk_file_1: Path, fdb_file_1: Path):
|
||||||
def test_1(db_1):
|
zipped_fbk_file = zipfile.Path(act_1.vars['files'] / 'core_5659.zip',
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
at='core_5659.fbk')
|
||||||
|
fbk_file_1.write_bytes(zipped_fbk_file.read_bytes())
|
||||||
|
#
|
||||||
|
with act_1.connect_server() as srv:
|
||||||
|
srv.database.restore(backup=fbk_file_1, database=fdb_file_1)
|
||||||
|
srv.wait()
|
||||||
|
#
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.isql(switches=['-q', f'localhost:{fdb_file_1}'], input=test_script_1, connect_db=False)
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -35,7 +35,7 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 3.0.3
|
# version: 3.0.3
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -112,7 +112,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
# con.close()
|
# con.close()
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
Error while executing SQL statement:
|
Error while executing SQL statement:
|
||||||
@ -131,8 +132,5 @@ expected_stdout_1 = """
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.3')
|
@pytest.mark.version('>=3.0.3')
|
||||||
@pytest.mark.xfail
|
|
||||||
def test_1(db_1):
|
def test_1(db_1):
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
pytest.skip("Requires encryption plugin")
|
||||||
|
|
||||||
|
|
||||||
|
@ -29,7 +29,7 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 2.5.8
|
# version: 2.5.8
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -76,7 +76,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
# con.close()
|
# con.close()
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
Try revert bytes in decimal value: 1 using struct format: "b" ; result: 01
|
Try revert bytes in decimal value: 1 using struct format: "b" ; result: 01
|
||||||
@ -95,8 +96,5 @@ expected_stdout_1 = """
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
@pytest.mark.version('>=2.5.8')
|
@pytest.mark.version('>=2.5.8')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action):
|
||||||
def test_1(db_1):
|
pytest.skip("Requires function not provided by driver")
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
#coding:utf-8
|
#coding:utf-8
|
||||||
#
|
#
|
||||||
# id: bugs.core_5685
|
# id: bugs.core_5685
|
||||||
# title: Sometime it is impossible to cancel\\kill connection executing external query
|
# title: Sometime it is impossible to cancel/kill connection executing external query
|
||||||
# decription:
|
# decription:
|
||||||
# Problem did appear when host "A" established connection to host "B" but could not get completed reply from this "B".
|
# Problem did appear when host "A" established connection to host "B" but could not get completed reply from this "B".
|
||||||
# This can be emulated by following steps:
|
# This can be emulated by following steps:
|
||||||
@ -54,12 +54,18 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
import re
|
||||||
|
import subprocess
|
||||||
|
import time
|
||||||
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
from firebird.driver import ShutdownMode, ShutdownMethod
|
||||||
|
|
||||||
# version: 3.0.3
|
# version: 3.0.3
|
||||||
# resources: None
|
# resources: None
|
||||||
|
|
||||||
substitutions_1 = [('.*After line.*', ''), ('.*Data source.*', '.*Data source'), ('.*HANGING_STATEMENT_BLOB_ID.*', '')]
|
substitutions_1 = [('.*After line.*', ''), ('.*Data source.*', '.*Data source'),
|
||||||
|
('.*HANGING_STATEMENT_BLOB_ID.*', '')]
|
||||||
|
|
||||||
init_script_1 = """
|
init_script_1 = """
|
||||||
create sequence g;
|
create sequence g;
|
||||||
@ -327,7 +333,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
HANGED ATTACH, STDOUT: Records affected: 0
|
HANGED ATTACH, STDOUT: Records affected: 0
|
||||||
@ -354,9 +361,89 @@ expected_stdout_1 = """
|
|||||||
KILLER ATTACH, STDOUT: Records affected: 0
|
KILLER ATTACH, STDOUT: Records affected: 0
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
kill_script = """
|
||||||
|
set list on;
|
||||||
|
set blob all;
|
||||||
|
select gen_id(g,1) as ITERATION_NO from rdb$database;
|
||||||
|
commit;
|
||||||
|
|
||||||
|
select
|
||||||
|
sign(a.mon$attachment_id) as hanging_attach_connection
|
||||||
|
,left(a.mon$remote_protocol,3) as hanging_attach_protocol
|
||||||
|
,s.mon$state as hanging_statement_state
|
||||||
|
,s.mon$sql_text as hanging_statement_blob_id
|
||||||
|
from rdb$database d
|
||||||
|
left join mon$attachments a on a.mon$remote_process containing 'isql'
|
||||||
|
-- do NOT use, field not existed in 2.5.x: and a.mon$system_flag is distinct from 1
|
||||||
|
and a.mon$attachment_id is distinct from current_connection
|
||||||
|
left join mon$statements s on
|
||||||
|
a.mon$attachment_id = s.mon$attachment_id
|
||||||
|
and s.mon$state = 1 -- 4.0 Classic: 'SELECT RDB$MAP_USING, RDB$MAP_PLUGIN, ... FROM RDB$AUTH_MAPPING', mon$state = 0
|
||||||
|
;
|
||||||
|
|
||||||
|
set count on;
|
||||||
|
delete from mon$attachments a
|
||||||
|
where
|
||||||
|
a.mon$attachment_id <> current_connection
|
||||||
|
and a.mon$remote_process containing 'isql'
|
||||||
|
;
|
||||||
|
commit;
|
||||||
|
"""
|
||||||
|
|
||||||
|
hang_script_1 = temp_file('hang_script.sql')
|
||||||
|
hang_stdout_1 = temp_file('hang_script.out')
|
||||||
|
hang_stderr_1 = temp_file('hang_script.err')
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.3')
|
@pytest.mark.version('>=3.0.3')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, hang_script_1: Path, hang_stdout_1: Path, hang_stderr_1: Path,
|
||||||
def test_1(db_1):
|
capsys):
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
hang_script_1.write_text('set list on; set count on; select * from sp_get_data;')
|
||||||
|
pattern_for_failed_statement = re.compile('Statement failed, SQLSTATE = (08006|08003)')
|
||||||
|
pattern_for_connection_close = re.compile('(Error (reading|writing) data (from|to) the connection)|(connection shutdown)')
|
||||||
|
pattern_for_ignored_messages = re.compile('(-send_packet/send)|(-Killed by database administrator.)')
|
||||||
|
killer_output = []
|
||||||
|
#
|
||||||
|
with open(hang_stdout_1, mode='w') as hang_out, open(hang_stderr_1, mode='w') as hang_err:
|
||||||
|
p_hang_sql = subprocess.Popen([act_1.vars['isql'], '-i', str(hang_script_1),
|
||||||
|
'-user', act_1.db.user,
|
||||||
|
'-password', act_1.db.password, act_1.db.dsn],
|
||||||
|
stdout=hang_out, stderr=hang_err)
|
||||||
|
try:
|
||||||
|
time.sleep(4)
|
||||||
|
for i in range(2):
|
||||||
|
act_1.reset()
|
||||||
|
act_1.isql(switches=[], input=kill_script)
|
||||||
|
killer_output.append(act_1.stdout)
|
||||||
|
finally:
|
||||||
|
p_hang_sql.terminate()
|
||||||
|
# Ensure that database is not busy
|
||||||
|
with act_1.connect_server() as srv:
|
||||||
|
srv.database.shutdown(database=act_1.db.db_path, mode=ShutdownMode.FULL,
|
||||||
|
method=ShutdownMethod.FORCED, timeout=0)
|
||||||
|
srv.database.bring_online(database=act_1.db.db_path)
|
||||||
|
#
|
||||||
|
output = []
|
||||||
|
for line in hang_stdout_1.read_text().splitlines():
|
||||||
|
if line.strip():
|
||||||
|
output.append(f'HANGED ATTACH, STDOUT: {line}')
|
||||||
|
for line in hang_stderr_1.read_text().splitlines():
|
||||||
|
if line.strip():
|
||||||
|
if pattern_for_ignored_messages.search(line):
|
||||||
|
continue
|
||||||
|
elif pattern_for_failed_statement.search(line):
|
||||||
|
msg = '<found pattern about failed statement>'
|
||||||
|
elif pattern_for_connection_close.search(line):
|
||||||
|
msg = '<found pattern about closed connection>'
|
||||||
|
else:
|
||||||
|
msg = line
|
||||||
|
output.append(f'HANGED ATTACH, STDERR: {msg}')
|
||||||
|
for step in killer_output:
|
||||||
|
for line in step.splitlines():
|
||||||
|
if line.strip():
|
||||||
|
output.append(f"KILLER ATTACH, STDOUT: {' '.join(line.split())}")
|
||||||
|
# Check
|
||||||
|
act_1.reset()
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.stdout = '\n'.join(output)
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -20,7 +20,11 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
import subprocess
|
||||||
|
import time
|
||||||
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
from firebird.driver import ShutdownMode, ShutdownMethod
|
||||||
|
|
||||||
# version: 3.0.3
|
# version: 3.0.3
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -211,23 +215,82 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
STDOUT in ISQL: CHECK_EDS_RESULT 1
|
CHECK_EDS_RESULT 1
|
||||||
STDOUT in ISQL: Records affected: 1
|
Records affected: 1
|
||||||
STDOUT in ISQL: Records affected: 0
|
Records affected: 0
|
||||||
STDOUT in ISQL: CHECK_EDS_RESULT 1
|
CHECK_EDS_RESULT 1
|
||||||
STDOUT in ISQL: Records affected: 1
|
Records affected: 1
|
||||||
STDOUT in ISQL: Records affected: 0
|
Records affected: 0
|
||||||
STDOUT in DB reset: Point before DB shutdown.
|
|
||||||
STDOUT in DB reset: Point after DB shutdown.
|
|
||||||
STDOUT in DB reset: Point after DB online.
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
eds_script = temp_file('eds_script.sql')
|
||||||
|
eds_output = temp_file('eds_script.out')
|
||||||
|
new_diff_file = temp_file('_new_diff_5704.tmp')
|
||||||
|
new_main_file = temp_file('new_main_5704.tmp')
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.3')
|
@pytest.mark.version('>=3.0.3')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, eds_script: Path, eds_output: Path, new_diff_file: Path,
|
||||||
def test_1(db_1):
|
new_main_file: Path):
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
eds_script.write_text(f"""
|
||||||
|
set count on;
|
||||||
|
set list on;
|
||||||
|
set autoddl off;
|
||||||
|
|
||||||
|
set term ^;
|
||||||
|
create or alter procedure sp_connect returns(check_eds_result int) as
|
||||||
|
declare usr varchar(31);
|
||||||
|
declare pwd varchar(31);
|
||||||
|
declare v_sttm varchar(255) = 'select 1 from rdb$database';
|
||||||
|
begin
|
||||||
|
usr ='{act_1.db.user}';
|
||||||
|
pwd = '{act_1.db.password}';
|
||||||
|
execute statement v_sttm
|
||||||
|
on external 'localhost:' || rdb$get_context('SYSTEM','DB_NAME')
|
||||||
|
as user usr password pwd
|
||||||
|
into check_eds_result;
|
||||||
|
suspend;
|
||||||
|
end
|
||||||
|
^
|
||||||
|
set term ^;
|
||||||
|
|
||||||
|
commit;
|
||||||
|
set transaction read committed no record_version lock timeout 1;
|
||||||
|
|
||||||
|
alter database add difference file '{new_diff_file}';
|
||||||
|
select * from sp_connect;
|
||||||
|
|
||||||
|
rollback;
|
||||||
|
select * from rdb$files;
|
||||||
|
rollback;
|
||||||
|
|
||||||
|
set transaction read committed no record_version lock timeout 1;
|
||||||
|
|
||||||
|
alter database add file '{new_main_file}';
|
||||||
|
select * from sp_connect;
|
||||||
|
--select * from rdb$files;
|
||||||
|
rollback;
|
||||||
|
select * from rdb$files;
|
||||||
|
""")
|
||||||
|
#
|
||||||
|
with open(eds_output, mode='w') as eds_out:
|
||||||
|
p_eds_sql = subprocess.Popen([act_1.vars['isql'], '-i', str(eds_script),
|
||||||
|
'-user', act_1.db.user,
|
||||||
|
'-password', act_1.db.password, act_1.db.dsn],
|
||||||
|
stdout=eds_out, stderr=subprocess.STDOUT)
|
||||||
|
try:
|
||||||
|
time.sleep(4)
|
||||||
|
finally:
|
||||||
|
p_eds_sql.terminate()
|
||||||
|
# Ensure that database is not busy
|
||||||
|
with act_1.connect_server() as srv:
|
||||||
|
srv.database.shutdown(database=act_1.db.db_path, mode=ShutdownMode.FULL,
|
||||||
|
method=ShutdownMethod.FORCED, timeout=0)
|
||||||
|
srv.database.bring_online(database=act_1.db.db_path)
|
||||||
|
# Check
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.stdout = eds_output.read_text()
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -41,7 +41,8 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from difflib import unified_diff
|
||||||
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 3.0.3
|
# version: 3.0.3
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -265,12 +266,30 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
|
trace_conf = """
|
||||||
|
# ::: NOTE :::
|
||||||
|
# First 'database' section here INTENTIONALLY was written WRONG!
|
||||||
|
database = (%[\\\\/](security[[:digit:]]).fdb|(security.db))
|
||||||
|
enabled = false
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
database =
|
||||||
|
{
|
||||||
|
enabled = true
|
||||||
|
log_connections = true
|
||||||
|
}
|
||||||
|
"""
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.3')
|
@pytest.mark.version('>=3.0.3')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action):
|
||||||
def test_1(db_1):
|
log_before = act_1.get_firebird_log()
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
with act_1.trace(config=trace_conf, keep_log=False):
|
||||||
|
# We run here ISQL only in order to "wake up" trace session and force it to raise error in its log.
|
||||||
|
# NO message like 'Statement failed, SQLSTATE = 08004/connection rejected by remote interface' should appear now!
|
||||||
|
act_1.isql(switches=['-n', '-q'], input='quit;')
|
||||||
|
log_after = act_1.get_firebird_log()
|
||||||
|
assert list(unified_diff(log_before, log_after)) == []
|
||||||
|
@ -18,7 +18,11 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
import zipfile
|
||||||
|
from difflib import unified_diff
|
||||||
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
from firebird.driver import SrvRestoreFlag
|
||||||
|
|
||||||
# version: 3.0
|
# version: 3.0
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -181,12 +185,27 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
# cleanup( f_list )
|
# cleanup( f_list )
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
|
fbk_file_1 = temp_file('core5719-ods-11_2.fbk')
|
||||||
|
fdb_file_1 = temp_file('check_restored_5719.fdb')
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0')
|
@pytest.mark.version('>=3.0')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, fbk_file_1: Path, fdb_file_1: Path):
|
||||||
def test_1(db_1):
|
zipped_fbk_file = zipfile.Path(act_1.vars['files'] / 'core_5719-ods-11_2.zip',
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
at='core5719-ods-11_2.fbk')
|
||||||
|
fbk_file_1.write_bytes(zipped_fbk_file.read_bytes())
|
||||||
|
log_before = act_1.get_firebird_log()
|
||||||
|
#
|
||||||
|
with act_1.connect_server() as srv:
|
||||||
|
srv.database.restore(backup=fbk_file_1, database=fdb_file_1,
|
||||||
|
flags=SrvRestoreFlag.REPLACE, verbose=True)
|
||||||
|
restore_err = [line for line in srv if 'ERROR' in line.upper()]
|
||||||
|
log_after = act_1.get_firebird_log()
|
||||||
|
srv.database.validate(database=fdb_file_1)
|
||||||
|
validate_err = [line for line in srv if 'ERROR' in line.upper()]
|
||||||
|
#
|
||||||
|
assert restore_err == []
|
||||||
|
assert validate_err == []
|
||||||
|
assert list(unified_diff(log_before, log_after)) == []
|
||||||
|
@ -23,7 +23,10 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
import subprocess
|
||||||
|
import time
|
||||||
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
|
||||||
# version: 3.0.3
|
# version: 3.0.3
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -130,16 +133,37 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
# cleanup( (f_show_command_sql, f_show_command_log, f_show_command_err) )
|
# cleanup( (f_show_command_sql, f_show_command_log, f_show_command_err) )
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
STDOUT: TEST1
|
TEST1
|
||||||
STDOUT: TEST1_ID_PK_DESC UNIQUE DESCENDING INDEX ON TEST1(ID)
|
TEST1_ID_PK_DESC UNIQUE DESCENDING INDEX ON TEST1(ID)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
show_script_1 = temp_file('show_script.sql')
|
||||||
|
show_output_1 = temp_file('show_script.out')
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.3')
|
@pytest.mark.version('>=3.0.3')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, show_script_1: Path, show_output_1: Path):
|
||||||
def test_1(db_1):
|
show_script_1.write_text('show table;show index;')
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
with act_1.db.connect() as con:
|
||||||
|
con.execute_immediate('recreate table test1(id int primary key using descending index test1_id_pk_desc)')
|
||||||
|
con.commit()
|
||||||
|
c = con.cursor()
|
||||||
|
c.execute('recreate table test2(id int primary key using descending index test2_id_pk_desc)')
|
||||||
|
# WARNING: we launch ISQL here in async mode in order to have ability to kill its
|
||||||
|
# process if it will hang!
|
||||||
|
with open(show_output_1, mode='w') as show_out:
|
||||||
|
p_show_sql = subprocess.Popen([act_1.vars['isql'], '-i', str(show_script_1),
|
||||||
|
'-user', act_1.db.user,
|
||||||
|
'-password', act_1.db.password, act_1.db.dsn],
|
||||||
|
stdout=show_out, stderr=subprocess.STDOUT)
|
||||||
|
try:
|
||||||
|
time.sleep(4)
|
||||||
|
finally:
|
||||||
|
p_show_sql.terminate()
|
||||||
|
#
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.stdout = show_output_1.read_text()
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -11,12 +11,16 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from difflib import unified_diff
|
||||||
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
|
||||||
# version: 4.0
|
# version: 4.0
|
||||||
# resources: None
|
# resources: None
|
||||||
|
|
||||||
substitutions_1 = []
|
substitutions_1 = [('database .*tmp_core_5771.fdb already exists.', 'database tmp_core_5771.fdb already exists.'),
|
||||||
|
('opened file .*tmp_core_5771.fbk',
|
||||||
|
'opened file tmp_core_5771.fbk')]
|
||||||
|
|
||||||
init_script_1 = """"""
|
init_script_1 = """"""
|
||||||
|
|
||||||
@ -151,17 +155,35 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
# cleanup( (f_restore_log,f_restore_err,f_fblog_before,f_fblog_after,f_diff_txt,tmpfbk,tmpfdb) )
|
# cleanup( (f_restore_log,f_restore_err,f_fblog_before,f_fblog_after,f_diff_txt,tmpfbk,tmpfdb) )
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
RESTORE STDOUT:GBAK:OPENED FILE TMP_CORE_5771.FBK
|
gbak:opened file tmp_core_5771.fbk
|
||||||
RESTORE STDERR: DATABASE TMP_5771_RESTORED.FDB ALREADY EXISTS. TO REPLACE IT, USE THE -REP SWITCH
|
|
||||||
RESTORE STDERR: -EXITING BEFORE COMPLETION DUE TO ERRORS
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
expected_stderr_1 = """
|
||||||
|
database tmp_core_5771.fdb already exists. To replace it, use the -REP switch
|
||||||
|
-Exiting before completion due to errors
|
||||||
|
"""
|
||||||
|
|
||||||
|
fbk_file = temp_file('tmp_core_5771.fbk')
|
||||||
|
fdb_file = temp_file('tmp_core_5771.fdb')
|
||||||
|
|
||||||
@pytest.mark.version('>=4.0')
|
@pytest.mark.version('>=4.0')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, fbk_file: Path, fdb_file: Path):
|
||||||
def test_1(db_1):
|
act_1.gbak(switches=['-b', act_1.db.dsn, str(fbk_file)])
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
act_1.gbak(switches=['-rep', str(fbk_file), f'localhost:{fdb_file}'])
|
||||||
|
#
|
||||||
|
log_before = act_1.get_firebird_log()
|
||||||
|
#
|
||||||
|
act_1.reset()
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.expected_stderr = expected_stderr_1
|
||||||
|
act_1.svcmgr(switches=['action_restore', 'bkp_file', str(fbk_file),
|
||||||
|
'dbname', str(fdb_file), 'verbose'])
|
||||||
|
#
|
||||||
|
log_after = act_1.get_firebird_log()
|
||||||
|
assert list(unified_diff(log_before, log_after)) == []
|
||||||
|
assert act_1.clean_stderr == act_1.clean_expected_stderr
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -40,7 +40,7 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 3.0.4
|
# version: 3.0.4
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -75,7 +75,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
Query line: select
|
Query line: select
|
||||||
@ -88,8 +89,16 @@ expected_stdout_1 = """
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.4')
|
@pytest.mark.version('>=3.0.4')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, capsys):
|
||||||
def test_1(db_1):
|
with act_1.db.connect() as con:
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
c = con.cursor()
|
||||||
|
sql_expr = "select \r -- comment N1 \r 'foo' as msg \r from \r -- comment N2 \r rdb$database"
|
||||||
|
for line in sql_expr.split('\r'):
|
||||||
|
print(f'Query line: {line}')
|
||||||
|
c.execute(sql_expr)
|
||||||
|
for row in c:
|
||||||
|
print(f'Query result: {row[0]}')
|
||||||
|
#
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.stdout = capsys.readouterr().out
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -18,7 +18,8 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, user_factory, User, temp_file
|
||||||
|
|
||||||
# version: 3.0.4
|
# version: 3.0.4
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -153,7 +154,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
# cleanup( f_list )
|
# cleanup( f_list )
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
RDB$USER TMP$C5790
|
RDB$USER TMP$C5790
|
||||||
@ -168,9 +170,38 @@ expected_stdout_1 = """
|
|||||||
Records affected: 0
|
Records affected: 0
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
test_user = user_factory(name='tmp$c5790', password='123')
|
||||||
|
fdb_file = temp_file('tmp_5790.fdb')
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.4')
|
@pytest.mark.version('>=3.0.4')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, test_user: User, fdb_file: Path):
|
||||||
def test_1(db_1):
|
test_script = f"""
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
create database 'localhost:{fdb_file}';
|
||||||
|
alter database set linger to 0;
|
||||||
|
commit;
|
||||||
|
grant drop database to {test_user.name};
|
||||||
|
commit;
|
||||||
|
connect 'localhost:{fdb_file}' user {test_user.name} password '{test_user.password}';
|
||||||
|
set list on;
|
||||||
|
set count on;
|
||||||
|
select
|
||||||
|
r.rdb$user -- {test_user.name}
|
||||||
|
,r.rdb$grantor -- sysdba
|
||||||
|
,r.rdb$privilege -- o
|
||||||
|
,r.rdb$grant_option -- 0
|
||||||
|
,r.rdb$relation_name -- sql$database
|
||||||
|
,r.rdb$field_name -- <null>
|
||||||
|
,r.rdb$user_type -- 8
|
||||||
|
,iif( r.rdb$object_type = decode( left(rdb$get_context('SYSTEM', 'ENGINE_VERSION'),1), '3',20, '4',21), 1, 0) "rdb_object_type_is_expected ?"
|
||||||
|
from rdb$user_privileges r
|
||||||
|
where r.rdb$user=upper('{test_user.name}');
|
||||||
|
|
||||||
|
-- this should NOT show any attachments: "Records affected: 0" must be shown here.
|
||||||
|
select * from mon$attachments where mon$attachment_id != current_connection;
|
||||||
|
commit;
|
||||||
|
|
||||||
|
drop database;
|
||||||
|
rollback;
|
||||||
|
"""
|
||||||
|
act_1.isql(switches=['-q'], input=test_script)
|
||||||
|
assert not fdb_file.exists()
|
||||||
|
@ -51,7 +51,7 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 3.0.4
|
# version: 3.0.4
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -151,7 +151,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
# db_conn.close()
|
# db_conn.close()
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
1.1. Trying to encrypt with existing key.
|
1.1. Trying to encrypt with existing key.
|
||||||
@ -170,8 +171,7 @@ expected_stdout_1 = """
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.4')
|
@pytest.mark.version('>=3.0.4')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action):
|
||||||
def test_1(db_1):
|
pytest.skip("Requires encryption plugin")
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -35,12 +35,14 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 3.0.4
|
# version: 3.0.4
|
||||||
# resources: None
|
# resources: None
|
||||||
|
|
||||||
substitutions_1 = [('ATTRIBUTES .* ENCRYPTED, PLUGIN .*', 'ATTRIBUTES ENCRYPTED'), ('CRYPT CHECKSUM.*', 'CRYPT CHECKSUM'), ('KEY HASH.*', 'KEY HASH'), ('ENCRYPTION KEY NAME.*', 'ENCRYPTION KEY')]
|
substitutions_1 = [('ATTRIBUTES .* ENCRYPTED, PLUGIN .*', 'ATTRIBUTES ENCRYPTED'),
|
||||||
|
('CRYPT CHECKSUM.*', 'CRYPT CHECKSUM'), ('KEY HASH.*', 'KEY HASH'),
|
||||||
|
('ENCRYPTION KEY NAME.*', 'ENCRYPTION KEY')]
|
||||||
|
|
||||||
init_script_1 = """"""
|
init_script_1 = """"""
|
||||||
|
|
||||||
@ -176,7 +178,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
ATTRIBUTES FORCE WRITE, ENCRYPTED, PLUGIN DBCRYPT
|
ATTRIBUTES FORCE WRITE, ENCRYPTED, PLUGIN DBCRYPT
|
||||||
@ -186,8 +189,7 @@ expected_stdout_1 = """
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.4')
|
@pytest.mark.version('>=3.0.4')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action):
|
||||||
def test_1(db_1):
|
pytest.skip("Requires encryption plugin")
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -34,7 +34,8 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
|
||||||
# version: 3.0.6
|
# version: 3.0.6
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -141,18 +142,39 @@ db_1 = db_factory(charset='WIN1251', sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
|
||||||
expected_stdout_1 = """
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
|
expected_stderr_1 = """
|
||||||
Statement failed, SQLSTATE = 42000
|
Statement failed, SQLSTATE = 42000
|
||||||
Dynamic SQL Error
|
Dynamic SQL Error
|
||||||
-SQL error code = -104
|
-SQL error code = -104
|
||||||
-Name longer than database column size
|
-Name longer than database column size
|
||||||
"""
|
"""
|
||||||
|
test_script = temp_file('test_script.sql')
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.6')
|
@pytest.mark.version('>=3.0.6')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, test_script: Path):
|
||||||
def test_1(db_1):
|
if act_1.is_version('<4'):
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
# Maximal number of characters in the column for FB 3.x is 31.
|
||||||
|
# Here we use name of 32 characters and this must raise error
|
||||||
|
# with text "Name longer than database column size":
|
||||||
|
column_title = 'СъешьЖеЕщёЭтихМягкихФранкоБулок'
|
||||||
|
else:
|
||||||
|
# Maximal number of characters in the column for FB 4.x is 63.
|
||||||
|
# Here we use name of 64 characters and this must raise error
|
||||||
|
# with text "Name longer than database column size":
|
||||||
|
column_title = 'СъешьЖеЕщёЭтихПрекрасныхФранкоБулокВместоДурацкихМорковныхКотлет'
|
||||||
|
# Code to be executed further in separate ISQL process:
|
||||||
|
test_script.write_text(f"""
|
||||||
|
set list on;
|
||||||
|
set sqlda_display on;
|
||||||
|
-- Maximal number of characters in the column for FB 3.x is 31.
|
||||||
|
-- Here we use name of 32 characters and this must raise error
|
||||||
|
-- with text "Name longer than database column size":
|
||||||
|
select 1 as "{column_title}" from rdb$database;
|
||||||
|
""", encoding='cp1251')
|
||||||
|
#
|
||||||
|
act_1.expected_stderr = expected_stderr_1
|
||||||
|
act_1.isql(switches=['-q'], input_file=test_script, charset='WIN1251')
|
||||||
|
assert act_1.clean_stderr == act_1.clean_expected_stderr
|
||||||
|
@ -34,7 +34,7 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 4.0
|
# version: 4.0
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -233,7 +233,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
+ VALIDATION STARTED
|
+ VALIDATION STARTED
|
||||||
@ -241,8 +242,5 @@ expected_stdout_1 = """
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
@pytest.mark.version('>=4.0')
|
@pytest.mark.version('>=4.0')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action):
|
||||||
def test_1(db_1):
|
pytest.skip("Requires encryption plugin")
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -41,12 +41,13 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 3.0.4
|
# version: 3.0.4
|
||||||
# resources: None
|
# resources: None
|
||||||
|
|
||||||
substitutions_1 = [('ATTRIBUTES FORCE WRITE, ENCRYPTED, PLUGIN.*', 'ATTRIBUTES FORCE WRITE, ENCRYPTED'), ('CRYPT CHECKSUM.*', 'CRYPT CHECKSUM'), ('KEY HASH.*', 'KEY HASH')]
|
substitutions_1 = [('ATTRIBUTES FORCE WRITE, ENCRYPTED, PLUGIN.*', 'ATTRIBUTES FORCE WRITE, ENCRYPTED'),
|
||||||
|
('CRYPT CHECKSUM.*', 'CRYPT CHECKSUM'), ('KEY HASH.*', 'KEY HASH')]
|
||||||
|
|
||||||
init_script_1 = """"""
|
init_script_1 = """"""
|
||||||
|
|
||||||
@ -176,7 +177,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
ATTRIBUTES FORCE WRITE, ENCRYPTED, PLUGIN DBCRYPT
|
ATTRIBUTES FORCE WRITE, ENCRYPTED, PLUGIN DBCRYPT
|
||||||
@ -186,8 +188,7 @@ expected_stdout_1 = """
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.4')
|
@pytest.mark.version('>=3.0.4')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action):
|
||||||
def test_1(db_1):
|
pytest.skip("Requires encryption plugin")
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
#coding:utf-8
|
#coding:utf-8
|
||||||
#
|
#
|
||||||
# id: bugs.core_5833
|
# id: bugs.core_5833
|
||||||
# title: Server crashes on preparing empty query when trace is enabled
|
# title: DDL triggers for some object types (views, exceptions, roles, indexes, domains) are lost in backup-restore process
|
||||||
# decription:
|
# decription:
|
||||||
# We create DDL triggers for all cases that are enumerated in $FB_HOME/doc/sql.extensions/README.ddl_triggers.txt.
|
# We create DDL triggers for all cases that are enumerated in $FB_HOME/doc/sql.extensions/README.ddl_triggers.txt.
|
||||||
# Then query to RDB$TRIGGERS table is applied to database and its results are stored in <log_file_1>.
|
# Then query to RDB$TRIGGERS table is applied to database and its results are stored in <log_file_1>.
|
||||||
@ -21,7 +21,9 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from difflib import unified_diff
|
||||||
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
|
||||||
# version: 3.0
|
# version: 3.0
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -218,12 +220,64 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
# cleanup( (f_ddl_sql, f_chk_sql, f_xmeta1_log, f_xmeta1_err, f_xmeta2_log, f_xmeta2_err, f_diff_txt, tmp_bkup, tmp_rest) )
|
# cleanup( (f_ddl_sql, f_chk_sql, f_xmeta1_log, f_xmeta1_err, f_xmeta2_log, f_xmeta2_err, f_diff_txt, tmp_bkup, tmp_rest) )
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
|
ddl_list = ['CREATE TABLE', 'ALTER TABLE', 'DROP TABLE',
|
||||||
|
'CREATE PROCEDURE', 'ALTER PROCEDURE', 'DROP PROCEDURE',
|
||||||
|
'CREATE FUNCTION', 'ALTER FUNCTION', 'DROP FUNCTION',
|
||||||
|
'CREATE TRIGGER', 'ALTER TRIGGER', 'DROP TRIGGER',
|
||||||
|
'CREATE EXCEPTION', 'ALTER EXCEPTION', 'DROP EXCEPTION',
|
||||||
|
'CREATE VIEW', 'ALTER VIEW', 'DROP VIEW',
|
||||||
|
'CREATE DOMAIN', 'ALTER DOMAIN', 'DROP DOMAIN',
|
||||||
|
'CREATE ROLE', 'ALTER ROLE', 'DROP ROLE',
|
||||||
|
'CREATE SEQUENCE', 'ALTER SEQUENCE', 'DROP SEQUENCE',
|
||||||
|
'CREATE USER', 'ALTER USER', 'DROP USER',
|
||||||
|
'CREATE INDEX', 'ALTER INDEX', 'DROP INDEX',
|
||||||
|
'CREATE COLLATION', 'DROP COLLATION', 'ALTER CHARACTER SET',
|
||||||
|
'CREATE PACKAGE', 'ALTER PACKAGE', 'DROP PACKAGE',
|
||||||
|
'CREATE PACKAGE BODY', 'DROP PACKAGE BODY']
|
||||||
|
|
||||||
|
test_script_1 = """
|
||||||
|
set blob all;
|
||||||
|
set list on;
|
||||||
|
set count on;
|
||||||
|
select rdb$trigger_name, rdb$trigger_type, rdb$trigger_source as blob_id_for_trg_source, rdb$trigger_blr as blob_id_for_trg_blr
|
||||||
|
from rdb$triggers
|
||||||
|
where rdb$system_flag is distinct from 1
|
||||||
|
order by 1;
|
||||||
|
"""
|
||||||
|
|
||||||
|
fbk_file = temp_file('tmp_5833.fbk')
|
||||||
|
fdb_file = temp_file('tmp_5833.fdb')
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0')
|
@pytest.mark.version('>=3.0')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, fbk_file: Path, fdb_file: Path):
|
||||||
def test_1(db_1):
|
script = ['set bail on;', 'set term ^;']
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
# Initial DDL: create all triggers
|
||||||
|
for item in ddl_list:
|
||||||
|
for evt_time in ['before', 'after']:
|
||||||
|
script.append(f"recreate trigger trg_{evt_time}_{item.replace(' ', '_').lower()} active {evt_time} {item.lower()} as")
|
||||||
|
script.append(" declare c rdb$field_name;")
|
||||||
|
script.append("begin")
|
||||||
|
script.append(" c = rdb$get_context('DDL_TRIGGER', 'OBJECT_NAME');")
|
||||||
|
script.append("end ^")
|
||||||
|
script.append("")
|
||||||
|
script.append("set term ;^")
|
||||||
|
script.append("commit;")
|
||||||
|
act_1.isql(switches=[], input='\n'.join(script))
|
||||||
|
# Query RDB$TRIGGERS before b/r:
|
||||||
|
act_1.reset()
|
||||||
|
act_1.isql(switches=[], input=test_script_1)
|
||||||
|
meta_before = [line for line in act_1.stdout.splitlines() if not line.startswith('BLOB_ID_FOR_TRG')]
|
||||||
|
# B/S
|
||||||
|
act_1.reset()
|
||||||
|
act_1.gbak(switches=['-b', act_1.db.dsn, str(fbk_file)])
|
||||||
|
act_1.reset()
|
||||||
|
act_1.gbak(switches=['-c', str(fbk_file), f'localhost:{fdb_file}'])
|
||||||
|
# Query RDB$TRIGGERS after b/r:
|
||||||
|
act_1.reset()
|
||||||
|
act_1.isql(switches=[f'localhost:{fdb_file}'], input=test_script_1, connect_db=False)
|
||||||
|
meta_after = [line for line in act_1.stdout.splitlines() if not line.startswith('BLOB_ID_FOR_TRG')]
|
||||||
|
# Check
|
||||||
|
assert list(unified_diff(meta_before, meta_after)) == []
|
||||||
|
@ -14,7 +14,7 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
|
||||||
# version: 3.0.3
|
# version: 3.0.3
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -76,17 +76,35 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
|
||||||
expected_stdout_1 = """
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
con1.rollback: point before.
|
|
||||||
con1.rollback: point after.
|
#expected_stdout_1 = """
|
||||||
sample-2 finished OK.
|
#con1.rollback: point before.
|
||||||
"""
|
#con1.rollback: point after.
|
||||||
|
#sample-2 finished OK.
|
||||||
|
#"""
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0.3')
|
@pytest.mark.version('>=3.0.3')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action):
|
||||||
def test_1(db_1):
|
with act_1.db.connect() as con1, act_1.db.connect() as con2:
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
c2 = con2.cursor()
|
||||||
|
# Following 'select count' is MANDATORY for reproduce:
|
||||||
|
c2.execute('select count(*) from gtt').fetchall()
|
||||||
|
#
|
||||||
|
c1 = con1.cursor()
|
||||||
|
c1.execute('insert into gtt(id) values(?)', [1])
|
||||||
|
c1.execute('insert into gtt(id) values(?)', [1])
|
||||||
|
#
|
||||||
|
c2.execute('insert into gtt(id) values(?)', [2])
|
||||||
|
#
|
||||||
|
con1.rollback()
|
||||||
|
#
|
||||||
|
c2.execute('insert into gtt(id) select 2 from rdb$types rows 200', [2])
|
||||||
|
con2.commit()
|
||||||
|
#
|
||||||
|
c1.execute('insert into gtt(id) values(?)', [11])
|
||||||
|
c1.execute('insert into gtt(id) values(?)', [11])
|
||||||
|
#
|
||||||
|
con1.rollback()
|
||||||
|
# This test does not need to assert anything, it passes if we get here without error
|
||||||
|
@ -15,12 +15,14 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action
|
||||||
|
from firebird.driver import DatabaseError
|
||||||
|
|
||||||
# version: 2.5.9
|
# version: 2.5.9
|
||||||
# resources: None
|
# resources: None
|
||||||
|
|
||||||
substitutions_1 = [('Problematic key value is .*', 'Problematic key value is')]
|
#substitutions_1 = [('Problematic key value is .*', 'Problematic key value is')]
|
||||||
|
substitutions_1 = []
|
||||||
|
|
||||||
init_script_1 = """
|
init_script_1 = """
|
||||||
recreate table test(
|
recreate table test(
|
||||||
@ -63,27 +65,31 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
# con2.close()
|
# con2.close()
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
|
||||||
expected_stdout_1 = """
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
1 0 : Error while executing SQL statement:
|
|
||||||
- SQLCODE: -803
|
|
||||||
- violation of PRIMARY or UNIQUE KEY constraint "TEST_UID_PK" on table "TEST"
|
|
||||||
- Problematic key value is ("UID" = x'AA70F788EB634073AD328C284F775A3E')
|
|
||||||
1 1 : -803
|
|
||||||
1 2 : 335544665
|
|
||||||
|
|
||||||
2 0 : Error while executing SQL statement:
|
#expected_stdout_1 = """
|
||||||
- SQLCODE: -803
|
#1 0 : Error while executing SQL statement:
|
||||||
- violation of PRIMARY or UNIQUE KEY constraint "TEST_UID_PK" on table "TEST"
|
#- SQLCODE: -803
|
||||||
- Problematic key value is ("UID" = x'AA70F788EB634073AD328C284F775A3E')
|
#- violation of PRIMARY or UNIQUE KEY constraint "TEST_UID_PK" on table "TEST"
|
||||||
2 1 : -803
|
#- Problematic key value is ("UID" = x'AA70F788EB634073AD328C284F775A3E')
|
||||||
2 2 : 335544665
|
#1 1 : -803
|
||||||
"""
|
#1 2 : 335544665
|
||||||
|
|
||||||
|
#2 0 : Error while executing SQL statement:
|
||||||
|
#- SQLCODE: -803
|
||||||
|
#- violation of PRIMARY or UNIQUE KEY constraint "TEST_UID_PK" on table "TEST"
|
||||||
|
#- Problematic key value is ("UID" = x'AA70F788EB634073AD328C284F775A3E')
|
||||||
|
#2 1 : -803
|
||||||
|
#2 2 : 335544665
|
||||||
|
#"""
|
||||||
|
|
||||||
@pytest.mark.version('>=2.5.9')
|
@pytest.mark.version('>=2.5.9')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action):
|
||||||
def test_1(db_1):
|
with act_1.db.connect(charset='utf8') as con1, act_1.db.connect() as con2:
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
c1 = con1.cursor()
|
||||||
|
c2 = con2.cursor()
|
||||||
|
cmd = 'insert into test(uid) select uid from test rows 1'
|
||||||
|
for c in [c1, c2]:
|
||||||
|
with pytest.raises(DatabaseError, match='.*Problematic key value is.*'):
|
||||||
|
c.execute(cmd)
|
||||||
|
@ -20,7 +20,8 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from pathlib import Path
|
||||||
|
from firebird.qa import db_factory, python_act, Action, temp_file
|
||||||
|
|
||||||
# version: 3.0
|
# version: 3.0
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -163,7 +164,8 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
SEQ_NAME new sequence
|
SEQ_NAME new sequence
|
||||||
@ -172,9 +174,29 @@ expected_stdout_1 = """
|
|||||||
foo rio bar
|
foo rio bar
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
test_script_1 = """
|
||||||
|
set list on;
|
||||||
|
set blob all;
|
||||||
|
set list on;
|
||||||
|
select
|
||||||
|
rdb$generator_name as seq_name,
|
||||||
|
rdb$initial_value as seq_init,
|
||||||
|
rdb$generator_increment as seq_incr,
|
||||||
|
rdb$description as blob_id
|
||||||
|
from rdb$generators
|
||||||
|
where rdb$system_flag is distinct from 1;
|
||||||
|
"""
|
||||||
|
|
||||||
|
fbk_file = temp_file('tmp_core_5855.fbk')
|
||||||
|
fdb_file = temp_file('tmp_core_5855.fdb')
|
||||||
|
|
||||||
@pytest.mark.version('>=3.0')
|
@pytest.mark.version('>=3.0')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, fbk_file: Path, fdb_file: Path):
|
||||||
def test_1(db_1):
|
with act_1.connect_server() as srv:
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
srv.database.backup(database=act_1.db.db_path, backup=fbk_file)
|
||||||
|
srv.wait()
|
||||||
|
srv.database.restore(backup=fbk_file, database=fdb_file)
|
||||||
|
srv.wait()
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.isql(switches=[f'localhost:{fdb_file}'], input=test_script_1, connect_db=False)
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
@ -19,7 +19,7 @@
|
|||||||
# qmid: None
|
# qmid: None
|
||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
from firebird.qa import db_factory, isql_act, Action
|
from firebird.qa import db_factory, python_act, Action, user_factory, User
|
||||||
|
|
||||||
# version: 4.0
|
# version: 4.0
|
||||||
# resources: None
|
# resources: None
|
||||||
@ -126,25 +126,83 @@ db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|||||||
#
|
#
|
||||||
#
|
#
|
||||||
#---
|
#---
|
||||||
#act_1 = python_act('db_1', test_script_1, substitutions=substitutions_1)
|
|
||||||
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
||||||
|
|
||||||
expected_stdout_1 = """
|
expected_stdout_1 = """
|
||||||
definer_-_who_am_i : TMP$C5892
|
definer_-_who_am_i TMP$C5892
|
||||||
definer_-_who_else_here : SYSDBA
|
definer_-_who_else_here SYSDBA
|
||||||
definer_-_effective_user : SYSDBA
|
definer_-_effective_user SYSDBA
|
||||||
|
|
||||||
definer_-_who_am_i : TMP$C5892
|
definer_-_who_am_i TMP$C5892
|
||||||
definer_-_who_else_here : TMP$C5892
|
definer_-_who_else_here TMP$C5892
|
||||||
definer_-_effective_user : SYSDBA
|
definer_-_effective_user SYSDBA
|
||||||
|
|
||||||
invoker_-_who_am_i : TMP$C5892
|
invoker_-_who_am_i TMP$C5892
|
||||||
invoker_-_who_else_here : TMP$C5892
|
invoker_-_who_else_here TMP$C5892
|
||||||
invoker_-_effective_user : TMP$C5892
|
invoker_-_effective_user TMP$C5892
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
sp_definer_ddl = """
|
||||||
|
create or alter procedure sp_test_definer returns( another_name varchar(31), another_conn_id int, execution_context varchar(31) ) SQL SECURITY DEFINER
|
||||||
|
as
|
||||||
|
begin
|
||||||
|
execution_context = rdb$get_context('SYSTEM', 'EFFECTIVE_USER');
|
||||||
|
for
|
||||||
|
select mon$user, mon$attachment_id
|
||||||
|
from mon$attachments a
|
||||||
|
where a.mon$system_flag is distinct from 1 and a.mon$attachment_id != current_connection
|
||||||
|
into
|
||||||
|
another_name,
|
||||||
|
another_conn_id
|
||||||
|
do suspend;
|
||||||
|
end
|
||||||
|
"""
|
||||||
|
|
||||||
|
sp_invoker_ddl = """
|
||||||
|
create or alter procedure sp_test_invoker returns( another_name varchar(31), another_conn_id int, execution_context varchar(31) ) SQL SECURITY INVOKER
|
||||||
|
as
|
||||||
|
begin
|
||||||
|
execution_context = rdb$get_context('SYSTEM', 'EFFECTIVE_USER');
|
||||||
|
for
|
||||||
|
select mon$user, mon$attachment_id
|
||||||
|
from mon$attachments a
|
||||||
|
where
|
||||||
|
a.mon$system_flag is distinct from 1
|
||||||
|
and a.mon$attachment_id != current_connection
|
||||||
|
and a.mon$user = current_user
|
||||||
|
into
|
||||||
|
another_name,
|
||||||
|
another_conn_id
|
||||||
|
do suspend;
|
||||||
|
end
|
||||||
|
"""
|
||||||
|
|
||||||
|
test_user = user_factory(name='TMP$C5892', password='123')
|
||||||
|
|
||||||
@pytest.mark.version('>=4.0')
|
@pytest.mark.version('>=4.0')
|
||||||
@pytest.mark.xfail
|
def test_1(act_1: Action, test_user: User, capsys):
|
||||||
def test_1(db_1):
|
sql_chk_definer = 'select current_user as "definer_-_who_am_i", d.another_name as "definer_-_who_else_here", d.execution_context as "definer_-_effective_user" from rdb$database r left join sp_test_definer d on 1=1'
|
||||||
pytest.fail("Test not IMPLEMENTED")
|
sql_chk_invoker = 'select current_user as "invoker_-_who_am_i", d.another_name as "invoker_-_who_else_here", d.execution_context as "invoker_-_effective_user" from rdb$database r left join sp_test_invoker d on 1=1'
|
||||||
|
with act_1.db.connect() as con1, \
|
||||||
|
act_1.db.connect(user=test_user.name, password=test_user.password) as con2, \
|
||||||
|
act_1.db.connect(user=test_user.name, password=test_user.password) as con3:
|
||||||
|
#
|
||||||
|
con1.execute_immediate(sp_definer_ddl)
|
||||||
|
con1.execute_immediate(sp_invoker_ddl)
|
||||||
|
con1.commit()
|
||||||
|
con1.execute_immediate('grant execute on procedure sp_test_definer to public')
|
||||||
|
con1.execute_immediate('grant execute on procedure sp_test_invoker to public')
|
||||||
|
con1.commit()
|
||||||
|
#
|
||||||
|
with con2.cursor() as c2:
|
||||||
|
c2.execute(sql_chk_definer)
|
||||||
|
act_1.print_data_list(c2)
|
||||||
|
#
|
||||||
|
with con2.cursor() as c2:
|
||||||
|
c2.execute(sql_chk_invoker)
|
||||||
|
act_1.print_data_list(c2)
|
||||||
|
# Check
|
||||||
|
act_1.expected_stdout = expected_stdout_1
|
||||||
|
act_1.stdout = capsys.readouterr().out
|
||||||
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|
||||||
|
Loading…
Reference in New Issue
Block a user