2018-09-17 23:37:02 +03:00
# Tests for process restarting in the pre-fork process model
#
# Copyright (C) Andrew Bartlett <abartlet@samba.org> 2018
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
""" Tests process restarting in the pre-fork process model.
NOTE : As this test kills samba processes it won ' t play nicely with other
tests , so needs to be run in it ' s own environment.
"""
import os
import signal
import time
import samba
from samba . tests import TestCase , delete_force
from samba . dcerpc import echo , netlogon
from samba . messaging import Messaging
from samba . samdb import SamDB
from samba . credentials import Credentials , DONT_USE_KERBEROS
2020-09-11 23:29:46 +03:00
from samba . common import get_string
2018-09-17 23:37:02 +03:00
from samba . dsdb import (
UF_WORKSTATION_TRUST_ACCOUNT ,
UF_PASSWD_NOTREQD )
from samba . dcerpc . misc import SEC_CHAN_WKSTA
from samba . auth import system_session
NUM_WORKERS = 4
MACHINE_NAME = " PFRS "
class PreforkProcessRestartTests ( TestCase ) :
def setUp ( self ) :
2023-11-28 06:38:22 +03:00
super ( ) . setUp ( )
2018-09-17 23:37:02 +03:00
lp_ctx = self . get_loadparm ( )
self . msg_ctx = Messaging ( lp_ctx = lp_ctx )
def get_process_data ( self ) :
services = self . msg_ctx . irpc_all_servers ( )
processes = [ ]
for service in services :
for id in service . ids :
processes . append ( ( service . name , id . pid ) )
return processes
def get_process ( self , name ) :
processes = self . get_process_data ( )
for pname , pid in processes :
if name == pname :
return pid
return None
def get_worker_pids ( self , name , workers ) :
pids = [ ]
for x in range ( workers ) :
process_name = " prefork-worker- {0} - {1} " . format ( name , x )
pids . append ( self . get_process ( process_name ) )
self . assertIsNotNone ( pids [ x ] )
return pids
def wait_for_workers ( self , name , workers ) :
num_workers = len ( workers )
for x in range ( num_workers ) :
process_name = " prefork-worker- {0} - {1} " . format ( name , x )
self . wait_for_process ( process_name , workers [ x ] , 0 , 1 , 30 )
def wait_for_process ( self , name , pid , initial_delay , wait , timeout ) :
time . sleep ( initial_delay )
delay = initial_delay
while delay < timeout :
p = self . get_process ( name )
if p is not None and p != pid :
# process has restarted
return
time . sleep ( wait )
delay + = wait
self . fail ( " Times out after {0} seconds waiting for {1} to restart " .
format ( delay , name ) )
def check_for_duplicate_processes ( self ) :
processes = self . get_process_data ( )
process_map = { }
for name , p in processes :
if ( name . startswith ( " prefork- " ) or
name . endswith ( " _server " ) or
name . endswith ( " srv " ) ) :
if name in process_map :
if p != process_map [ name ] :
self . fail (
" Duplicate process for {0} , pids {1} and {2} " .
format ( name , p , process_map [ name ] ) )
def simple_bind ( self ) :
creds = self . insta_creds ( template = self . get_credentials ( ) )
creds . set_bind_dn ( " %s \\ %s " % ( creds . get_domain ( ) ,
creds . get_username ( ) ) )
self . samdb = SamDB ( url = " ldaps:// %s " % os . environ [ " SERVER " ] ,
lp = self . get_loadparm ( ) ,
credentials = creds )
def rpc_echo ( self ) :
conn = echo . rpcecho ( " ncalrpc: " , self . get_loadparm ( ) )
2020-02-07 01:02:38 +03:00
self . assertEqual ( [ 1 , 2 , 3 ] , conn . EchoData ( [ 1 , 2 , 3 ] ) )
2018-09-17 23:37:02 +03:00
def netlogon ( self ) :
server = os . environ [ " SERVER " ]
host = os . environ [ " SERVER_IP " ]
lp = self . get_loadparm ( )
credentials = self . get_credentials ( )
session = system_session ( )
ldb = SamDB ( url = " ldap:// %s " % host ,
session_info = session ,
credentials = credentials ,
lp = lp )
machine_pass = samba . generate_random_password ( 32 , 32 )
machine_name = MACHINE_NAME
machine_dn = " cn= %s , %s " % ( machine_name , ldb . domain_dn ( ) )
delete_force ( ldb , machine_dn )
utf16pw = ( ' " %s " ' % get_string ( machine_pass ) ) . encode ( ' utf-16-le ' )
ldb . add ( {
" dn " : machine_dn ,
" objectclass " : " computer " ,
" sAMAccountName " : " %s $ " % machine_name ,
" userAccountControl " :
str ( UF_WORKSTATION_TRUST_ACCOUNT | UF_PASSWD_NOTREQD ) ,
" unicodePwd " : utf16pw } )
machine_creds = Credentials ( )
machine_creds . guess ( lp )
machine_creds . set_secure_channel_type ( SEC_CHAN_WKSTA )
machine_creds . set_kerberos_state ( DONT_USE_KERBEROS )
machine_creds . set_password ( machine_pass )
machine_creds . set_username ( machine_name + " $ " )
machine_creds . set_workstation ( machine_name )
netlogon . netlogon (
" ncacn_ip_tcp: %s [schannel,seal] " % server ,
lp ,
machine_creds )
delete_force ( ldb , machine_dn )
def test_ldap_master_restart ( self ) :
# check ldap connection, do a simple bind
self . simple_bind ( )
# get ldap master process
pid = self . get_process ( " prefork-master-ldap " )
self . assertIsNotNone ( pid )
# Get the worker processes
workers = self . get_worker_pids ( " ldap " , NUM_WORKERS )
# kill it
os . kill ( pid , signal . SIGTERM )
# wait for the process to restart
self . wait_for_process ( " prefork-master-ldap " , pid , 1 , 1 , 30 )
# restarting the master restarts the workers as well, so make sure
# they have finished restarting
self . wait_for_workers ( " ldap " , workers )
# get ldap master process
new_pid = self . get_process ( " prefork-master-ldap " )
self . assertIsNotNone ( new_pid )
# check that the pid has changed
2023-01-18 22:37:03 +03:00
self . assertNotEqual ( pid , new_pid )
2018-09-17 23:37:02 +03:00
# check that the worker processes have restarted
new_workers = self . get_worker_pids ( " ldap " , NUM_WORKERS )
for x in range ( NUM_WORKERS ) :
2023-01-18 22:37:03 +03:00
self . assertNotEqual ( workers [ x ] , new_workers [ x ] )
2018-09-17 23:37:02 +03:00
# check that the previous server entries have been removed.
self . check_for_duplicate_processes ( )
# check ldap connection, another simple bind
self . simple_bind ( )
def test_ldap_worker_restart ( self ) :
# check ldap connection, do a simple bind
self . simple_bind ( )
# get ldap master process
pid = self . get_process ( " prefork-master-ldap " )
self . assertIsNotNone ( pid )
# Get the worker processes
workers = self . get_worker_pids ( " ldap " , NUM_WORKERS )
# kill worker 0
os . kill ( workers [ 0 ] , signal . SIGTERM )
# wait for the process to restart
self . wait_for_process ( " prefork-worker-ldap-0 " , pid , 1 , 1 , 30 )
# get ldap master process
new_pid = self . get_process ( " prefork-master-ldap " )
self . assertIsNotNone ( new_pid )
# check that the pid has not changed
2020-02-07 01:02:38 +03:00
self . assertEqual ( pid , new_pid )
2018-09-17 23:37:02 +03:00
# check that the worker processes have restarted
new_workers = self . get_worker_pids ( " ldap " , NUM_WORKERS )
# process 0 should have a new pid the others should be unchanged
2023-01-18 22:37:03 +03:00
self . assertNotEqual ( workers [ 0 ] , new_workers [ 0 ] )
2020-02-07 01:02:38 +03:00
self . assertEqual ( workers [ 1 ] , new_workers [ 1 ] )
self . assertEqual ( workers [ 2 ] , new_workers [ 2 ] )
self . assertEqual ( workers [ 3 ] , new_workers [ 3 ] )
2018-09-17 23:37:02 +03:00
# check that the previous server entries have been removed.
self . check_for_duplicate_processes ( )
# check ldap connection, another simple bind
self . simple_bind ( )
#
# Kill all the ldap worker processes and ensure that they are restarted
# correctly
#
def test_ldap_all_workers_restart ( self ) :
# check ldap connection, do a simple bind
self . simple_bind ( )
# get ldap master process
pid = self . get_process ( " prefork-master-ldap " )
self . assertIsNotNone ( pid )
# Get the worker processes
workers = self . get_worker_pids ( " ldap " , NUM_WORKERS )
# kill all the worker processes
for x in workers :
os . kill ( x , signal . SIGTERM )
# wait for the worker processes to restart
self . wait_for_workers ( " ldap " , workers )
# get ldap master process
new_pid = self . get_process ( " prefork-master-ldap " )
self . assertIsNotNone ( new_pid )
# check that the pid has not changed
2020-02-07 01:02:38 +03:00
self . assertEqual ( pid , new_pid )
2018-09-17 23:37:02 +03:00
# check that the worker processes have restarted
new_workers = self . get_worker_pids ( " ldap " , NUM_WORKERS )
for x in range ( NUM_WORKERS ) :
2023-01-18 22:37:03 +03:00
self . assertNotEqual ( workers [ x ] , new_workers [ x ] )
2018-09-17 23:37:02 +03:00
# check that the previous server entries have been removed.
self . check_for_duplicate_processes ( )
# check ldap connection, another simple bind
self . simple_bind ( )
def test_rpc_master_restart ( self ) :
# check rpc connection, make a rpc echo request
self . rpc_echo ( )
# get rpc master process
pid = self . get_process ( " prefork-master-rpc " )
self . assertIsNotNone ( pid )
# Get the worker processes
workers = self . get_worker_pids ( " rpc " , NUM_WORKERS )
# kill it
os . kill ( pid , signal . SIGTERM )
# wait for the process to restart
self . wait_for_process ( " prefork-master-rpc " , pid , 1 , 1 , 30 )
# wait for workers to restart as well
self . wait_for_workers ( " rpc " , workers )
# get ldap master process
new_pid = self . get_process ( " prefork-master-rpc " )
self . assertIsNotNone ( new_pid )
# check that the pid has changed
2023-01-18 22:37:03 +03:00
self . assertNotEqual ( pid , new_pid )
2018-09-17 23:37:02 +03:00
# check that the worker processes have restarted
new_workers = self . get_worker_pids ( " rpc " , NUM_WORKERS )
for x in range ( NUM_WORKERS ) :
2023-01-18 22:37:03 +03:00
self . assertNotEqual ( workers [ x ] , new_workers [ x ] )
2018-09-17 23:37:02 +03:00
# check that the previous server entries have been removed.
self . check_for_duplicate_processes ( )
# check rpc connection, another rpc echo request
self . rpc_echo ( )
def test_rpc_worker_zero_restart ( self ) :
# check rpc connection, make a rpc echo request and a netlogon request
self . rpc_echo ( )
self . netlogon ( )
# get rpc master process
pid = self . get_process ( " prefork-master-rpc " )
self . assertIsNotNone ( pid )
# Get the worker processes
workers = self . get_worker_pids ( " rpc " , NUM_WORKERS )
# kill worker 0
os . kill ( workers [ 0 ] , signal . SIGTERM )
# wait for the process to restart
self . wait_for_process ( " prefork-worker-rpc-0 " , workers [ 0 ] , 1 , 1 , 30 )
# get rpc master process
new_pid = self . get_process ( " prefork-master-rpc " )
self . assertIsNotNone ( new_pid )
# check that the pid has not changed
2020-02-07 01:02:38 +03:00
self . assertEqual ( pid , new_pid )
2018-09-17 23:37:02 +03:00
# check that the worker processes have restarted
new_workers = self . get_worker_pids ( " rpc " , NUM_WORKERS )
# process 0 should have a new pid the others should be unchanged
2023-01-18 22:37:03 +03:00
self . assertNotEqual ( workers [ 0 ] , new_workers [ 0 ] )
2020-02-07 01:02:38 +03:00
self . assertEqual ( workers [ 1 ] , new_workers [ 1 ] )
self . assertEqual ( workers [ 2 ] , new_workers [ 2 ] )
self . assertEqual ( workers [ 3 ] , new_workers [ 3 ] )
2018-09-17 23:37:02 +03:00
# check that the previous server entries have been removed.
self . check_for_duplicate_processes ( )
# check rpc connection, another rpc echo request, and netlogon request
self . rpc_echo ( )
self . netlogon ( )
def test_rpc_all_workers_restart ( self ) :
# check rpc connection, make a rpc echo request, and a netlogon request
self . rpc_echo ( )
self . netlogon ( )
# get rpc master process
pid = self . get_process ( " prefork-master-rpc " )
self . assertIsNotNone ( pid )
# Get the worker processes
workers = self . get_worker_pids ( " rpc " , NUM_WORKERS )
# kill all the worker processes
for x in workers :
os . kill ( x , signal . SIGTERM )
# wait for the worker processes to restart
for x in range ( NUM_WORKERS ) :
self . wait_for_process (
" prefork-worker-rpc- {0} " . format ( x ) , workers [ x ] , 0 , 1 , 30 )
# get rpc master process
new_pid = self . get_process ( " prefork-master-rpc " )
self . assertIsNotNone ( new_pid )
# check that the pid has not changed
2020-02-07 01:02:38 +03:00
self . assertEqual ( pid , new_pid )
2018-09-17 23:37:02 +03:00
# check that the worker processes have restarted
new_workers = self . get_worker_pids ( " rpc " , NUM_WORKERS )
for x in range ( NUM_WORKERS ) :
2023-01-18 22:37:03 +03:00
self . assertNotEqual ( workers [ x ] , new_workers [ x ] )
2018-09-17 23:37:02 +03:00
# check that the previous server entries have been removed.
self . check_for_duplicate_processes ( )
# check rpc connection, another rpc echo request and netlogon
self . rpc_echo ( )
self . netlogon ( )
def test_master_restart_backoff ( self ) :
# get kdc master process
2019-07-11 00:28:30 +03:00
pid = self . get_process ( " prefork-master-echo " )
2018-09-17 23:37:02 +03:00
self . assertIsNotNone ( pid )
#
# Check that the processes get backed off as expected
#
# have prefork backoff increment = 5
# prefork maximum backoff = 10
backoff_increment = 5
for expected in [ 0 , 5 , 10 , 10 ] :
# Get the worker processes
workers = self . get_worker_pids ( " kdc " , NUM_WORKERS )
2019-07-11 00:28:30 +03:00
process = self . get_process ( " prefork-master-echo " )
2018-09-17 23:37:02 +03:00
os . kill ( process , signal . SIGTERM )
# wait for the process to restart
start = time . time ( )
2019-07-11 00:28:30 +03:00
self . wait_for_process ( " prefork-master-echo " , process , 0 , 1 , 30 )
2018-09-17 23:37:02 +03:00
# wait for the workers to restart as well
2019-07-11 00:28:30 +03:00
self . wait_for_workers ( " echo " , workers )
2018-09-17 23:37:02 +03:00
end = time . time ( )
duration = end - start
# process restart will take some time. Check that the elapsed
# duration falls somewhere in the expected range, i.e. we haven't
# taken longer than the backoff increment
self . assertLess ( duration , expected + backoff_increment )
self . assertGreaterEqual ( duration , expected )
# check that the worker processes have restarted
2019-07-11 00:28:30 +03:00
new_workers = self . get_worker_pids ( " echo " , NUM_WORKERS )
2018-09-17 23:37:02 +03:00
for x in range ( NUM_WORKERS ) :
2023-01-18 22:37:03 +03:00
self . assertNotEqual ( workers [ x ] , new_workers [ x ] )
2018-09-17 23:37:02 +03:00
# check that the previous server entries have been removed.
self . check_for_duplicate_processes ( )
def test_worker_restart_backoff ( self ) :
#
# Check that the processes get backed off as expected
#
# have prefork backoff increment = 5
# prefork maximum backoff = 10
backoff_increment = 5
for expected in [ 0 , 5 , 10 , 10 ] :
2019-07-11 00:28:30 +03:00
process = self . get_process ( " prefork-worker-echo-2 " )
2018-09-17 23:37:02 +03:00
self . assertIsNotNone ( process )
os . kill ( process , signal . SIGTERM )
# wait for the process to restart
start = time . time ( )
2019-07-11 00:28:30 +03:00
self . wait_for_process ( " prefork-worker-echo-2 " , process , 0 , 1 , 30 )
2018-09-17 23:37:02 +03:00
end = time . time ( )
duration = end - start
# process restart will take some time. Check that the elapsed
# duration falls somewhere in the expected range, i.e. we haven't
# taken longer than the backoff increment
self . assertLess ( duration , expected + backoff_increment )
self . assertGreaterEqual ( duration , expected )
self . check_for_duplicate_processes ( )