2021-04-26 20:07:00 +02:00
|
|
|
#coding:utf-8
|
|
|
|
#
|
|
|
|
# id: bugs.core_6018
|
|
|
|
# title: Make it possible to start multiple transactions (possibly in different attachments) using the same initial transaction snapshot
|
2021-12-10 19:50:31 +01:00
|
|
|
# decription:
|
2021-04-26 20:07:00 +02:00
|
|
|
# We open first connect using FDB and set custom transaction parameter block which is used to start SNAPSHOT transaction.
|
|
|
|
# Within this first transaction (tx1a) we insert into test table record with value = -2 and commit this Tx.
|
|
|
|
# Then we do start next transaction (also SNAPSHOT; its name = 'tx1b') and obtain value of RDB$GET_CONTEXT('SYSTEM', 'SNAPSHOT_NUMBER').
|
2021-12-10 19:50:31 +01:00
|
|
|
# Also, in this second 'tx1b' we add one more record into table with value = -1 using autonomous transaction --> BOTH records should be seen
|
2021-04-26 20:07:00 +02:00
|
|
|
# in another attachment that will be started after this moment.
|
|
|
|
# But if this (new) attachment will start Tx with requirement to use snapshot that was for Tx1a then it must see only FIRST record with value=-2.
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# We launch then ISQL for establish another transaction and make it perform two transactions:
|
|
|
|
# 1) 'set transaction snapshot' --> must extract both records from test table
|
2021-12-10 19:50:31 +01:00
|
|
|
# === vs ===
|
2021-04-26 20:07:00 +02:00
|
|
|
# 2) 'set transaction snapshot at number %(snap_num)s' --> must extract only FIRST record with value = -2.
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# THis is done TWO times: when based snapshot is KNOWN (i.e. tx1b is alive) and after tx1b is committed and base is unknown.
|
|
|
|
# Second ISQL launch must issue error:
|
|
|
|
# Statement failed, SQLSTATE = 0B000
|
|
|
|
# Transaction's base snapshot number does not exist
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# Checked on: 4.0.0.1457 (SS,CS) -- works fine.
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# tracker_id: CORE-6018
|
|
|
|
# min_versions: ['4.0']
|
|
|
|
# versions: 4.0
|
|
|
|
# qmid: None
|
|
|
|
|
|
|
|
import pytest
|
2021-12-10 19:50:31 +01:00
|
|
|
from firebird.qa import db_factory, python_act, Action
|
|
|
|
from firebird.driver import tpb, Isolation
|
2021-04-26 20:07:00 +02:00
|
|
|
|
|
|
|
# version: 4.0
|
|
|
|
# resources: None
|
|
|
|
|
|
|
|
substitutions_1 = []
|
|
|
|
|
|
|
|
init_script_1 = """
|
2021-12-10 19:50:31 +01:00
|
|
|
recreate table tsn (sn int);
|
2021-04-26 20:07:00 +02:00
|
|
|
commit;
|
2021-12-10 19:50:31 +01:00
|
|
|
"""
|
2021-04-26 20:07:00 +02:00
|
|
|
|
|
|
|
db_1 = db_factory(sql_dialect=3, init=init_script_1)
|
|
|
|
|
|
|
|
# test_script_1
|
|
|
|
#---
|
|
|
|
# import os
|
|
|
|
# import re
|
|
|
|
# import time
|
|
|
|
# import subprocess
|
|
|
|
# from subprocess import Popen
|
|
|
|
# from fdb import services
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# os.environ["ISC_USER"] = user_name
|
|
|
|
# os.environ["ISC_PASSWORD"] = user_password
|
|
|
|
# db_conn.close()
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# customTPB = ( [ fdb.isc_tpb_concurrency, fdb.isc_tpb_nowait ] )
|
|
|
|
# con1 = fdb.connect( dsn = dsn )
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# tx1a=con1.trans( default_tpb = customTPB )
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# cur1=tx1a.cursor()
|
|
|
|
# cur1.execute('insert into tsn(sn) values( -2 )' )
|
|
|
|
# tx1a.commit()
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# sql_get_sn='''
|
|
|
|
# execute block returns(o_sn bigint) as
|
|
|
|
# begin
|
|
|
|
# o_sn = RDB$GET_CONTEXT('SYSTEM', 'SNAPSHOT_NUMBER');
|
|
|
|
# suspend;
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# in autonomous transaction do
|
|
|
|
# insert into tsn(sn) values( -1 );
|
|
|
|
# end
|
|
|
|
# '''
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# tx1b=con1.trans( default_tpb = customTPB )
|
|
|
|
# cur1=tx1b.cursor()
|
|
|
|
# cur1.execute( sql_get_sn )
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# snap_num = -1
|
|
|
|
# for r in cur1:
|
|
|
|
# snap_num = r[0]
|
|
|
|
# # print( r[0] )
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# for m in ('yet exists', 'does not exists'):
|
|
|
|
# sql_chk_sn='''
|
|
|
|
# -- NB!! looks strange but it seems that this 'SET BAIL ON' does not work here because
|
|
|
|
# -- both records will be extracted in any case. // todo later: check it!
|
|
|
|
# --set bail on;
|
2021-12-10 19:50:31 +01:00
|
|
|
# set count on;
|
|
|
|
# commit;
|
2021-04-26 20:07:00 +02:00
|
|
|
# set transaction snapshot;
|
|
|
|
# select 'Tx base snapshot: %(m)s' as msg, t.sn as set_tx_snapshot_without_num from tsn t order by sn;
|
|
|
|
# commit;
|
|
|
|
# set transaction snapshot at number %(snap_num)s;
|
|
|
|
# select 'Tx base snapshot: %(m)s' as msg, t.sn as set_tx_snapshot_at_number_N from tsn t order by sn;
|
2021-12-10 19:50:31 +01:00
|
|
|
# commit;
|
2021-04-26 20:07:00 +02:00
|
|
|
# quit;
|
|
|
|
# ''' % ( locals() )
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# #print(sql_chk_sn)
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# runProgram('isql', [ dsn, '-q' ], sql_chk_sn)
|
|
|
|
# if tx1b:
|
|
|
|
# tx1b.commit()
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
# cur1.close()
|
|
|
|
# con1.close()
|
2021-12-10 19:50:31 +01:00
|
|
|
#
|
|
|
|
#
|
2021-04-26 20:07:00 +02:00
|
|
|
#---
|
|
|
|
|
2021-12-10 19:50:31 +01:00
|
|
|
act_1 = python_act('db_1', substitutions=substitutions_1)
|
|
|
|
|
|
|
|
expected_stdout_1_a = """
|
2021-04-26 20:07:00 +02:00
|
|
|
MSG SET_TX_SNAPSHOT_WITHOUT_NUM
|
|
|
|
============================ ===========================
|
|
|
|
Tx base snapshot: yet exists -2
|
|
|
|
Tx base snapshot: yet exists -1
|
|
|
|
Records affected: 2
|
|
|
|
MSG SET_TX_SNAPSHOT_AT_NUMBER_N
|
|
|
|
============================ ===========================
|
|
|
|
Tx base snapshot: yet exists -2
|
|
|
|
Records affected: 1
|
2021-12-10 19:50:31 +01:00
|
|
|
"""
|
2021-04-26 20:07:00 +02:00
|
|
|
|
2021-12-10 19:50:31 +01:00
|
|
|
expected_stdout_1_b = """
|
2021-04-26 20:07:00 +02:00
|
|
|
MSG SET_TX_SNAPSHOT_WITHOUT_NUM
|
|
|
|
================================= ===========================
|
|
|
|
Tx base snapshot: does not exists -2
|
|
|
|
Tx base snapshot: does not exists -1
|
|
|
|
Records affected: 2
|
|
|
|
MSG SET_TX_SNAPSHOT_AT_NUMBER_N
|
|
|
|
================================= ===========================
|
|
|
|
Tx base snapshot: does not exists -2
|
|
|
|
Tx base snapshot: does not exists -1
|
|
|
|
Records affected: 2
|
2021-12-10 19:50:31 +01:00
|
|
|
"""
|
|
|
|
|
2021-04-26 20:07:00 +02:00
|
|
|
expected_stderr_1 = """
|
|
|
|
Statement failed, SQLSTATE = 0B000
|
|
|
|
Transaction's base snapshot number does not exist
|
2021-12-10 19:50:31 +01:00
|
|
|
"""
|
2021-04-26 20:07:00 +02:00
|
|
|
|
|
|
|
@pytest.mark.version('>=4.0')
|
2021-12-10 19:50:31 +01:00
|
|
|
def test_1(act_1: Action):
|
|
|
|
custom_tpb = tpb(isolation=Isolation.CONCURRENCY, lock_timeout=0)
|
|
|
|
with act_1.db.connect() as con1:
|
|
|
|
tx1a = con1.transaction_manager(custom_tpb)
|
|
|
|
tx1a.begin()
|
|
|
|
cur1 = tx1a.cursor()
|
|
|
|
cur1.execute('insert into tsn (sn) values( -2 )')
|
|
|
|
tx1a.commit()
|
|
|
|
#
|
|
|
|
sql_get_sn = """
|
|
|
|
execute block returns(o_sn bigint) as
|
|
|
|
begin
|
|
|
|
o_sn = RDB$GET_CONTEXT('SYSTEM', 'SNAPSHOT_NUMBER');
|
|
|
|
suspend;
|
2021-04-26 20:07:00 +02:00
|
|
|
|
2021-12-10 19:50:31 +01:00
|
|
|
in autonomous transaction do
|
|
|
|
insert into tsn(sn) values( -1 );
|
|
|
|
end
|
2021-12-22 20:23:11 +01:00
|
|
|
"""
|
2021-12-10 19:50:31 +01:00
|
|
|
tx1b = con1.transaction_manager(custom_tpb)
|
|
|
|
cur1 = tx1b.cursor()
|
|
|
|
snap_num = cur1.execute(sql_get_sn).fetchone()[0]
|
|
|
|
#
|
|
|
|
for msg, expect_out, expect_err in [('yet exists', expected_stdout_1_a, ''),
|
|
|
|
('does not exists', expected_stdout_1_b, expected_stderr_1)]:
|
|
|
|
sql_chk_sn = f"""
|
|
|
|
-- NB!! looks strange but it seems that this 'SET BAIL ON' does not work here because
|
|
|
|
-- both records will be extracted in any case. // todo later: check it!
|
|
|
|
--set bail on;
|
|
|
|
set count on;
|
|
|
|
commit;
|
|
|
|
set transaction snapshot;
|
|
|
|
select 'Tx base snapshot: {msg}' as msg, t.sn as set_tx_snapshot_without_num from tsn t order by sn;
|
|
|
|
commit;
|
|
|
|
set transaction snapshot at number {snap_num};
|
|
|
|
select 'Tx base snapshot: {msg}' as msg, t.sn as set_tx_snapshot_at_number_N from tsn t order by sn;
|
|
|
|
commit;
|
|
|
|
quit;
|
2021-12-22 20:23:11 +01:00
|
|
|
"""
|
2021-12-10 19:50:31 +01:00
|
|
|
act_1.reset()
|
|
|
|
act_1.expected_stdout = expect_out
|
|
|
|
act_1.expected_stderr = expect_err
|
|
|
|
act_1.isql(switches=['-q'], input=sql_chk_sn)
|
|
|
|
if tx1b.is_active():
|
|
|
|
tx1b.commit()
|
|
|
|
assert act_1.clean_stdout == act_1.clean_expected_stdout
|