6
0
mirror of https://github.com/FirebirdSQL/firebird-qa.git synced 2025-01-23 14:03:06 +01:00
firebird-qa/tests/bugs/gh_6782_test.py

351 lines
12 KiB
Python
Raw Normal View History

#coding:utf-8
2022-01-27 20:08:36 +01:00
"""
ID: issue-6782
ISSUE: 6782
TITLE: Getting "records fetched" for functions/procedures in trace
DESCRIPTION:
Confirmed bug on 4.0.0.2436: there was no lines with numner of:
* number of fetched rows;
* additional info about number of fetches/reads/writes/marks (after elapsed time).
In other words, trace log:
* was before fix:
Procedure <some_name>
0 ms
* became after fix:
Procedure <some_name>
5 records fetched <<< --- added
0 ms, 10 fetch(es)
^^^^^^^^^^^^ --- added
Test parses trace log and search there lines with names of known procedures/functions and
then checks presence of lines with number of fetched records (for selectable procedures) and
additional statistics ('fetches/reads/writes/marks').
Checked on:
5.0.0.87 SS: 7.231s.
5.0.0.85 CS: 6.425s.
4.0.1.2520 SS: 6.929s.
4.0.1.2519 CS: 9.452s.
3.0.8.33476 SS: 12.199s.
3.0.8.33476 CS: 14.090s.
NOTES:
[29.06.2021]
Added delay for 1.1 second after ISQL finished its script and before we ask trace to stop.
This is the only way to make trace log be completed. DO NOT EVER remove this delay because fbsvcmgr
makes query to FB services only one time per SECOND.
See also reply from Vlad related to test for core-6469, privately, letter: 04-mar-2021 13:02.
FBTEST: bugs.gh_6782
2022-01-27 20:08:36 +01:00
"""
import pytest
2022-01-27 20:08:36 +01:00
from firebird.qa import *
2022-01-27 20:08:36 +01:00
db = db_factory()
2022-01-27 20:08:36 +01:00
act = python_act('db')
2022-01-27 20:08:36 +01:00
expected_stdout = """
Procedure STANDALONE_SELECTABLE_SP:
FOUND line with number of fetched records
FOUND line with execution statistics
Function STANDALONE_FUNC:
FOUND line with execution statistics
Procedure STANDALONE_NONSELECTED_SP:
FOUND line with execution statistics
Procedure PG_TEST.PACKAGED_SELECTABLE_SP:
FOUND line with number of fetched records
FOUND line with execution statistics
Function PG_TEST.PACKAGED_FUNC:
FOUND line with execution statistics
Procedure PG_TEST.PACKAGED_NONSELECTED_SP:
FOUND line with execution statistics
Procedure SP_MAIN:
FOUND line with execution statistics
"""
2022-01-27 20:08:36 +01:00
@pytest.mark.skip('FIXME: Not IMPLEMENTED')
@pytest.mark.version('>=3.0.8')
def test_1(act: Action):
pytest.fail("Not IMPLEMENTED")
# test_script_1
#---
# import os
# import re
# import subprocess
# from subprocess import Popen
# import time
2022-01-25 22:55:48 +01:00
#
# os.environ["ISC_USER"] = user_name
# os.environ["ISC_PASSWORD"] = user_password
# db_conn.close()
2022-01-25 22:55:48 +01:00
#
# #--------------------------------------------
2022-01-25 22:55:48 +01:00
#
# def flush_and_close( file_handle ):
# # https://docs.python.org/2/library/os.html#os.fsync
# # If you're starting with a Python file object f,
# # first do f.flush(), and
# # then do os.fsync(f.fileno()), to ensure that all internal buffers associated with f are written to disk.
# global os
2022-01-25 22:55:48 +01:00
#
# file_handle.flush()
# if file_handle.mode not in ('r', 'rb') and file_handle.name != os.devnull:
# # otherwise: "OSError: [Errno 9] Bad file descriptor"!
# os.fsync(file_handle.fileno())
# file_handle.close()
2022-01-25 22:55:48 +01:00
#
# #--------------------------------------------
2022-01-25 22:55:48 +01:00
#
# def cleanup( f_names_list ):
# global os
# for i in range(len( f_names_list )):
# if type(f_names_list[i]) == file:
# del_name = f_names_list[i].name
# elif type(f_names_list[i]) == str:
# del_name = f_names_list[i]
# else:
# print('Unrecognized type of element:', f_names_list[i], ' - can not be treated as file.')
# print('type(f_names_list[i])=',type(f_names_list[i]))
# del_name = None
2022-01-25 22:55:48 +01:00
#
# if del_name and os.path.isfile( del_name ):
# os.remove( del_name )
2022-01-25 22:55:48 +01:00
#
# #--------------------------------------------
2022-01-25 22:55:48 +01:00
#
#
# # Prepare config for trace session that will be launched by call of FBSVCMGR:
# ################
# txt = '''database= %[\\\\\\\\/]bugs.gh_6782.fdb
# {
# enabled = true
2022-01-25 22:55:48 +01:00
# time_threshold = 0
# log_initfini = false
# log_errors = true
# log_procedure_finish = true
# log_function_finish = true
# }
# '''
# trc_cfg=open( os.path.join(context['temp_directory'],'tmp_trace_6782.cfg'), 'w')
# trc_cfg.write(txt)
# flush_and_close( trc_cfg )
2022-01-25 22:55:48 +01:00
#
# #####################################################################
# # Async. launch of trace session using FBSVCMGR action_trace_start:
2022-01-25 22:55:48 +01:00
#
# trc_log = open( os.path.join(context['temp_directory'],'tmp_trace_6782.log'), 'w')
2022-01-25 22:55:48 +01:00
#
# # Execute a child program in a new process, redirecting STDERR to the same target as of STDOUT:
# p_trace = Popen( [context['fbsvcmgr_path'], "localhost:service_mgr",
# "action_trace_start","trc_cfg", trc_cfg.name],
# stdout=trc_log, stderr=subprocess.STDOUT)
2022-01-25 22:55:48 +01:00
#
# # Wait! Trace session is initialized not instantly!
# time.sleep(1)
2022-01-25 22:55:48 +01:00
#
# #####################################################################
2022-01-25 22:55:48 +01:00
#
# # Determine active trace session ID (for further stop):
# trc_lst = open( os.path.join(context['temp_directory'],'tmp_trace_6782.lst'), 'w')
# subprocess.call([context['fbsvcmgr_path'], "localhost:service_mgr",
# "action_trace_list"],
# stdout=trc_lst, stderr=subprocess.STDOUT
# )
# flush_and_close( trc_lst )
2022-01-25 22:55:48 +01:00
#
# # Session ID: 5
# # ...
# trcssn = None
# with open( trc_lst.name,'r') as f:
# for line in f:
# if 'Session ID' in line and len(line.split())>=3:
# trcssn = line.split()[2]
# break
2022-01-25 22:55:48 +01:00
#
# # Result: `trcssn` is ID of active trace session.
# if not trcssn:
# print("Error parsing trace session ID.")
# flush_and_close( trc_log )
2022-01-25 22:55:48 +01:00
#
# else:
# #####################################################################
2022-01-25 22:55:48 +01:00
#
# # Preparing script for ISQL:
2022-01-25 22:55:48 +01:00
#
# sql_cmd='''
# create table test(id int);
# insert into test(id) select row_number()over() from rdb$types rows 5;
# commit;
# -- set echo on;
# set term ^;
# create procedure standalone_selectable_sp returns(id int) as
# begin
# for select id from test as cursor c
# do begin
# update test set id = -id * (select count(*) from rdb$database)
# where current of c;
# suspend;
# end
# end
# ^
2022-01-25 22:55:48 +01:00
#
# create procedure standalone_nonselected_sp as
# begin
# for select id from test as cursor c
# do begin
# update test set id = -id * (select count(*) from rdb$database)
# where current of c;
# end
# end
# ^
2022-01-25 22:55:48 +01:00
#
# create function standalone_func returns int as
# begin
# update test set id = rand()*10000000;
# return (select max(id) from test);
# end
# ^
2022-01-25 22:55:48 +01:00
#
# create package pg_test as
# begin
# procedure packaged_selectable_sp returns(id int);
# function packaged_func returns int;
# procedure packaged_nonselected_sp;
# end
# ^
2022-01-25 22:55:48 +01:00
#
# create package body pg_test as
# begin
# procedure packaged_selectable_sp returns(id int) as
# begin
# for select id from test as cursor c
# do begin
# update test set id = -id * (select count(*) from rdb$database)
# where current of c;
# suspend;
# end
# end
2022-01-25 22:55:48 +01:00
#
# procedure packaged_nonselected_sp as
# begin
# for select id from test as cursor c
# do begin
# update test set id = -id * (select count(*) from rdb$database)
# where current of c;
# end
# end
2022-01-25 22:55:48 +01:00
#
# function packaged_func returns int as
# begin
# update test set id = rand()*10000000;
# return (select min(id) from test);
# end
# end
# ^
2022-01-25 22:55:48 +01:00
#
#
# create procedure sp_main as
# declare c int;
# begin
2022-01-25 22:55:48 +01:00
# for select id from standalone_selectable_sp into c do
# begin
# -- nop --
# end
# ----------------------
# c = standalone_func();
# ----------------------
# execute procedure standalone_nonselected_sp;
# ----------------------
2022-01-25 22:55:48 +01:00
#
# for select id from pg_test.packaged_selectable_sp into c do
# begin
# -- nop --
# end
# ----------------------
# c = pg_test.packaged_func();
# ----------------------
2022-01-25 22:55:48 +01:00
#
# execute procedure pg_test.packaged_nonselected_sp;
# end
# ^
# set term ;^
# commit;
2022-01-25 22:55:48 +01:00
#
# set list on;
# execute procedure sp_main;
# commit;
# '''
2022-01-25 22:55:48 +01:00
#
# isql_cmd=open( os.path.join(context['temp_directory'],'tmp_isql_6782.sql'), 'w')
# isql_cmd.write(sql_cmd)
# flush_and_close( isql_cmd )
2022-01-25 22:55:48 +01:00
#
# #######################################################################
# isql_log=open( os.path.join(context['temp_directory'],'tmp_isql_6782.log'), 'w')
# subprocess.call( [ context['isql_path'], dsn, "-i", isql_cmd.name ], stdout=isql_log,stderr=subprocess.STDOUT)
# flush_and_close( isql_log )
2022-01-25 22:55:48 +01:00
#
# #########################
# ### C R U C I A L ###
# #########################
# # We must stay idle here at least 1.1 second before ask fbsbcmgr to stop trace.
# # The reason is that fbsvcmgr make query to FB services only one time per second.
# # If we do not take delay here then test can fail with (most often) empty log
# # or its log can look incompleted.
# # See letter from Vlad, 04-mar-2021 13:02 (subj: "Test core_6469 on Linux...").
# ###############
# time.sleep(1.1)
# ###############
2022-01-25 22:55:48 +01:00
#
# # Stop trace session. We have to do this BEFORE we termitane process with PID = p_trace.
# # NOTE ONCE AGAIN: we can do this only after delay for at least 1.1 second!
# trc_lst=open(trc_lst.name, "a")
# trc_lst.seek(0,2)
# subprocess.call([ context['fbsvcmgr_path'], "localhost:service_mgr",
# "action_trace_stop","trc_id",trcssn],
# stdout=trc_lst, stderr=subprocess.STDOUT
# )
# flush_and_close( trc_lst )
2022-01-25 22:55:48 +01:00
#
# ###########################
# p_trace.terminate()
# flush_and_close( trc_log )
# ###########################
2022-01-25 22:55:48 +01:00
#
# # Check logs:
# #############
# allowed_patterns = (
# re.compile('Procedure\\s+(STANDALONE_SELECTABLE_SP:|STANDALONE_NONSELECTED_SP:|PG_TEST.PACKAGED_SELECTABLE_SP:|PG_TEST.PACKAGED_NONSELECTED_SP:|SP_MAIN:)', re.IGNORECASE)
# ,re.compile('Function\\s+(STANDALONE_FUNC:|PG_TEST.PACKAGED_FUNC:)', re.IGNORECASE)
# ,re.compile('\\d+\\s+record(s|\\(s\\))?\\s+fetched', re.IGNORECASE)
# ,re.compile('\\s+\\d+\\s+ms(,)?\\s+\\d+\\s+fetch(es|\\(es\\))((,)?\\s+\\d+\\s+read(s|\\(s\\)))?((,)?\\s+\\d+\\s+write(s|\\(s\\)))?(,)?\\s+\\d+\\s+mark(s|\\(s\\))', re.IGNORECASE)
# )
2022-01-25 22:55:48 +01:00
#
# with open(trc_log.name) as f:
# for line in f:
# if line.split():
# match2some = filter( None, [ p.search(line) for p in allowed_patterns ] )
# if match2some:
# if ' ms' in line and 'fetch' in line:
# print('FOUND line with execution statistics')
# elif 'record' in line and 'fetch' in line:
# print('FOUND line with number of fetched records')
# else:
# print(line)
2022-01-25 22:55:48 +01:00
#
# # Cleanup:
# time.sleep(1)
# cleanup( (trc_lst, trc_cfg, trc_log, isql_cmd, isql_log) )
2022-01-25 22:55:48 +01:00
#
#---