6
0
mirror of https://github.com/FirebirdSQL/firebird-qa.git synced 2025-01-22 21:43:06 +01:00
firebird-qa/tests/bugs/core_3328_test.py

325 lines
12 KiB
Python
Raw Normal View History

2021-04-26 20:07:00 +02:00
#coding:utf-8
#
# id: bugs.core_3328
# title: Client writes error messages into firebird.log when database is shutted down
2021-11-17 19:43:06 +01:00
# decription:
2021-04-26 20:07:00 +02:00
# Test retrieves FB engine version in order to issue proper command option for getting firebird.log content
# (in 2.5 this is 'action_get_ib_log' rather then expected 'action_get_fb_log' - note on letter 'i' instead 'f').
# Initial content of firebird.log is saved into file, see var. 'f_fblog_before'.
# Then it starts child process (ISQL) which does some "heavy DML" activity and allows this ISQL to do it several seconds.
# After this, main thread calls FBSVCMGR with commands to move database to SHUTDOWN state and back to online.
# This should terminate child ISQL and after small delay one may to query new content of firebird.log.
# New firebird.log is saved into file, see var. 'f_fblog_after', so further its two versions can be compared.
# Comparison is done by using standard Python package 'difflib'.
# Difference between old and new firebird.log should _NOT_ contain lines with words 'gds__detach' or 'lost'.
# If these words absent - all fine, actual and expected output both have to be empty.
2021-11-17 19:43:06 +01:00
#
# 04-dec-2016.
2021-04-26 20:07:00 +02:00
# Checked on: WI-V2.5.7.27028, WI-V3.0.2.32642, WI-T4.0.0.460 (all on SS/SC/CS).
# Reduced time of ISQL working from 5 to 2 seconds.
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# Samples of call with '-c <path_to_fbclient_dll>':
2021-11-17 19:43:06 +01:00
#
# fbt_run -b C:\\MIX\\firebird\\fb25\\bin bugs.core_3328 -o localhost/3255 -c C:\\MIX\\firebird\\fb25\\fbinbclient.dll
# fbt_run -b C:\\MIX\\firebird\\fb25Cs\\bin bugs.core_3328 -o localhost/3249 -c C:\\MIX\\firebird\\fb25cs\\bin\\fbclient.dll
# fbt_run -b C:\\MIX\\firebird\\fb40Cs bugs.core_3328 -o localhost/3439 -c C:\\MIX\\firebird\\fb40cs\\fbclient.dll
# fbt_run -b C:\\MIX\\firebird\\fb40sc bugs.core_3328 -o localhost/3430 -c C:\\MIX\\firebird\\fb40sc\\fbclient.dll & fbt_view -d results.trf
#
2021-04-26 20:07:00 +02:00
# tracker_id: CORE-3328
# min_versions: ['2.5.1']
# versions: 2.5.1
# qmid: None
import pytest
2021-11-17 19:43:06 +01:00
import time
from difflib import unified_diff
from threading import Thread
from firebird.qa import db_factory, python_act, Action
from firebird.driver import ShutdownMethod, ShutdownMode
2021-04-26 20:07:00 +02:00
# version: 2.5.1
# resources: None
2021-11-17 19:43:06 +01:00
# substitutions_1 = [('attachments: [1-9]+', 'attachments: 1'), ('[\\s]+', ' ')]
substitutions_1 = [('database.*shutdown', 'database shutdown')]
2021-04-26 20:07:00 +02:00
init_script_1 = """
create table test(s varchar(36) unique);
commit;
2021-11-17 19:43:06 +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 subprocess
# from subprocess import Popen
# import time
# import difflib
# from fdb import services
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# os.environ["ISC_USER"] = user_name
# os.environ["ISC_PASSWORD"] = user_password
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# db_file=db_conn.database_name
# engine = str(db_conn.engine_version) # convert to text because 'float' object has no attribute 'startswith'
# db_conn.close()
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# #---------------------------------------------------
# def svc_get_fb_log( engine, f_fb_log ):
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# import subprocess
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# if engine.startswith('2.5'):
# get_firebird_log_key='action_get_ib_log'
# else:
# get_firebird_log_key='action_get_fb_log'
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# subprocess.call( [ context['fbsvcmgr_path'],
# "localhost:service_mgr",
# get_firebird_log_key
# ],
2021-11-17 19:43:06 +01:00
# stdout=f_fb_log,
2021-04-26 20:07:00 +02:00
# stderr=subprocess.STDOUT
# )
# return
2021-11-17 19:43:06 +01:00
#
#
2021-04-26 20:07:00 +02:00
# #---------------------------------------------
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# def flush_and_close(file_handle):
# # https://docs.python.org/2/library/os.html#os.fsync
2021-11-17 19:43:06 +01:00
# # If you're starting with a Python file object f,
# # first do f.flush(), and
2021-04-26 20:07:00 +02:00
# # then do os.fsync(f.fileno()), to ensure that all internal buffers associated with f are written to disk.
# global os
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# file_handle.flush()
# if file_handle.mode not in ('r', 'rb'):
# # otherwise: "OSError: [Errno 9] Bad file descriptor"!
# os.fsync(file_handle.fileno())
# file_handle.close()
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# #--------------------------------------------
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# def cleanup( f_names_list ):
# global os
# for i in range(len( f_names_list )):
# if os.path.isfile( f_names_list[i]):
# os.remove( f_names_list[i] )
# if os.path.isfile( f_names_list[i]):
# print('ERROR: can not remove file ' + f_names_list[i])
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# #--------------------------------------------
2021-11-17 19:43:06 +01:00
#
#
2021-04-26 20:07:00 +02:00
# # Get HOME dir of FB instance that is now checked.
2021-11-17 19:43:06 +01:00
# # We will concatenate this string with 'fbsvcmgr' command in order to choose
2021-04-26 20:07:00 +02:00
# # PROPER binary file when querying services API to shutdown/online DB
# # NOTE: this is NECESSARY if several instances are in work and we did not change
# # PATH variable before call this test!
2021-11-17 19:43:06 +01:00
#
# # NB, 06.12.2016: as of fdb 1.6.1 one need to EXPLICITLY specify user+password pair when doing connect
2021-04-26 20:07:00 +02:00
# # via to FB services API by services.connect() - see FB tracker, PYFB-69
# # ("Can not connect to FB services if set ISC_USER & ISC_PASSWORD by os.environ[ ... ]")
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# f_fblog_before=open( os.path.join(context['temp_directory'],'tmp_3328_fblog_before.txt'), 'w')
# svc_get_fb_log( engine, f_fblog_before )
# flush_and_close( f_fblog_before )
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# sql_dml='''
# show version;
# set term ^;
# execute block as
# declare v_role varchar(31);
# begin
# v_role = left(replace( uuid_to_char(gen_uuid()), '-', ''), 31);
# while (1=1) do
# begin
# insert into test(s) values( uuid_to_char( gen_uuid() ) );
# end
# end
# ^
# set term ;^
# '''
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# f_client_sql = open( os.path.join(context['temp_directory'],'tmp_client_3328.sql'), 'w')
# f_client_sql.write(sql_dml)
# flush_and_close( f_client_sql )
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# f_client_log = open( os.path.join(context['temp_directory'],'tmp_client_3328.log'), 'w')
# p_client_dml=subprocess.Popen( [context['isql_path'], dsn, "-n", "-i", f_client_sql.name ],
# stdout = f_client_log,
# stderr = subprocess.STDOUT
# )
# time.sleep(2)
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# f_shutdown_log = open( os.path.join(context['temp_directory'],'tmp_shutdown_and_online_3328.log'), 'w')
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# # Databases:
# # Number of attachments: 1
# # Number of databases: 1
# # Database in use: C:\\MIX\\FIREBIRD\\QA\\FBT-REPO\\TMP\\BUGS.CORE_3328.FDB
# subprocess.call( [context['fbsvcmgr_path'],"localhost:service_mgr",
# "info_svr_db_info",
# ],
# stdout = f_shutdown_log,
# stderr=subprocess.STDOUT
# )
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# subprocess.call( [context['fbsvcmgr_path'], "localhost:service_mgr",
# "action_properties", "prp_shutdown_mode", "prp_sm_full", "prp_shutdown_db", "0",
# "dbname", db_file,
# ],
# stdout = f_shutdown_log,
# stderr = subprocess.STDOUT
# )
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# # Databases:
# # Number of attachments: 0
# # Number of databases: 0
# subprocess.call( [context['fbsvcmgr_path'],"localhost:service_mgr",
# "info_svr_db_info",
# ],
# stdout = f_shutdown_log,
# stderr=subprocess.STDOUT
# )
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# subprocess.call( [context['fbsvcmgr_path'],"localhost:service_mgr",
# "action_db_stats",
# "dbname", db_file, "sts_hdr_pages"
# ],
# stdout = f_shutdown_log,
# stderr=subprocess.STDOUT
# )
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# subprocess.call( [context['fbsvcmgr_path'], "localhost:service_mgr",
# "action_properties", "prp_db_online",
# "dbname", db_file,
# ],
# stdout = f_shutdown_log,
# stderr = subprocess.STDOUT
# )
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# subprocess.call( [context['fbsvcmgr_path'],"localhost:service_mgr",
# "action_db_stats",
# "dbname", db_file, "sts_hdr_pages"
# ],
# stdout = f_shutdown_log,
# stderr=subprocess.STDOUT
# )
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# flush_and_close( f_shutdown_log )
2021-11-17 19:43:06 +01:00
#
#
2021-04-26 20:07:00 +02:00
# f_fblog_after=open( os.path.join(context['temp_directory'],'tmp_3328_fblog_after.txt'), 'w')
# svc_get_fb_log( engine, f_fblog_after )
# flush_and_close( f_fblog_after )
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# p_client_dml.terminate()
# flush_and_close( f_client_log )
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# # Now we can compare two versions of firebird.log and check their difference.
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# oldfb=open(f_fblog_before.name, 'r')
# newfb=open(f_fblog_after.name, 'r')
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# difftext = ''.join(difflib.unified_diff(
2021-11-17 19:43:06 +01:00
# oldfb.readlines(),
2021-04-26 20:07:00 +02:00
# newfb.readlines()
# ))
# oldfb.close()
# newfb.close()
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# f_diff_txt=open( os.path.join(context['temp_directory'],'tmp_3328_diff.txt'), 'w')
# f_diff_txt.write(difftext)
# flush_and_close( f_diff_txt )
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# # New lines in firebird.log must |NOT| contain these:
# # ===
# # REMOTE INTERFACE/gds__detach: Unsuccesful detach from database.
2021-11-17 19:43:06 +01:00
# # Uncommitted work may have been lost
2021-04-26 20:07:00 +02:00
# # ===
# # If such lines present - this is regression and we output them.
# # When all fine, final output is empty.
# #
# # BTW: for 3.0, firebird.log will contain such text:
# # INET/INET_ERROR: READ ERRNO = 10054, SERVER HOST = LOCALHOST, ADDRESS = 127.0.0.1/3333
# # -- but this is checked by another .fbt
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# with open( f_shutdown_log.name,'r') as f:
# for line in f:
# if ( 'unknown' in line.lower() or 'attributes' in line.lower() ):
# print(line)
2021-11-17 19:43:06 +01:00
# #or 'attachments' in line.lower()
#
2021-04-26 20:07:00 +02:00
# with open( f_diff_txt.name,'r') as f:
# for line in f:
# if line.startswith('+') and ('gds__detach' in line or 'lost' in line):
# print(line.upper())
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# ###############################
# # Cleanup.
# cleanup( [i.name for i in (f_shutdown_log,f_client_sql,f_client_log,f_fblog_before,f_fblog_after,f_diff_txt) ] )
2021-11-17 19:43:06 +01:00
#
2021-04-26 20:07:00 +02:00
# # NB: temply removed following lines from expected_stdout,
# # see core-5413, "fbsvcmgr info_svr_db_info does not see active attachments and databases in use (CLASSIC only)"
# # Number of attachments: 1
# # Number of attachments: 0
2021-11-17 19:43:06 +01:00
#
#
2021-04-26 20:07:00 +02:00
#---
2021-11-17 19:43:06 +01:00
act_1 = python_act('db_1', substitutions=substitutions_1)
2021-04-26 20:07:00 +02:00
2021-11-17 19:43:06 +01:00
expected_stderr_1 = """
Statement failed, SQLSTATE = HY000
database /tmp/pytest-of-pcisar/pytest-528/test_10/test.fdb shutdown
Statement failed, SQLSTATE = HY000
database /tmp/pytest-of-pcisar/pytest-528/test_10/test.fdb shutdown
"""
2021-04-26 20:07:00 +02:00
2021-11-17 19:43:06 +01:00
def run_work(act: Action):
test_script = """
show version;
set term ^;
execute block as
declare v_role varchar(31);
begin
v_role = left(replace( uuid_to_char(gen_uuid()), '-', ''), 31);
while (1=1) do
begin
insert into test(s) values( uuid_to_char( gen_uuid() ) );
end
end
^
set term ;^
2021-12-22 20:23:11 +01:00
"""
2021-11-17 19:43:06 +01:00
act.expected_stderr = expected_stderr_1
act.isql(switches=['-n'], input=test_script)
2021-04-26 20:07:00 +02:00
2021-11-17 19:43:06 +01:00
@pytest.mark.version('>=3.0')
def test_1(act_1: Action):
with act_1.connect_server() as srv:
srv.info.get_log()
log_before = srv.readlines()
#
work_thread = Thread(target=run_work, args=[act_1])
work_thread.start()
time.sleep(2)
#
srv.database.shutdown(database=act_1.db.db_path, mode=ShutdownMode.FULL,
2021-11-17 19:43:06 +01:00
method=ShutdownMethod.FORCED, timeout=0)
srv.database.bring_online(database=act_1.db.db_path)
2021-11-17 19:43:06 +01:00
#
srv.info.get_log()
log_after = srv.readlines()
#
work_thread.join(2)
if work_thread.is_alive():
pytest.fail('Work thread is still alive')
#
assert list(unified_diff(log_before, log_after)) == []
assert act_1.clean_stderr == act_1.clean_expected_stderr