@@ -879,11 +879,19 @@ def execute(
879
879
Specifies a timeout in seconds for the git command, after which the process
880
880
should be killed. This will have no effect if `as_process` is set to True.
881
881
It is set to None by default and will let the process run until the timeout
882
- is explicitly specified. This feature is not supported on Windows. It's also
883
- worth noting that `kill_after_timeout` uses SIGKILL, which can have negative
884
- side effects on a repository. For example, stale locks in case of ``git gc``
885
- could render the repository incapable of accepting changes until the lock is
886
- manually removed.
882
+ is explicitly specified. Uses of this feature should be carefully
883
+ considered, due to the following limitations:
884
+
885
+ 1. This feature is not supported at all on Windows.
886
+ 2. Effectiveness may vary by operating system. ``ps --ppid`` is used to
887
+ enumerate child processes, which is available on most GNU/Linux systems
888
+ but not most others.
889
+ 3. Deeper descendants do not receive signals, though they may sometimes
890
+ terminate as a consequence of their parent processes being killed.
891
+ 4. `kill_after_timeout` uses ``SIGKILL``, which can have negative side
892
+ effects on a repository. For example, stale locks in case of ``git gc``
893
+ could render the repository incapable of accepting changes until the lock
894
+ is manually removed.
887
895
888
896
:param with_stdout:
889
897
If True, default True, we open stdout on the created process.
@@ -1007,11 +1015,9 @@ def execute(
1007
1015
1008
1016
def kill_process (pid : int ) -> None :
1009
1017
"""Callback to kill a process."""
1010
- p = Popen (
1011
- ["ps" , "--ppid" , str (pid )],
1012
- stdout = PIPE ,
1013
- creationflags = PROC_CREATIONFLAGS ,
1014
- )
1018
+ if os .name == "nt" :
1019
+ raise AssertionError ("Bug: This callback would be ineffective and unsafe on Windows, stopping." )
1020
+ p = Popen (["ps" , "--ppid" , str (pid )], stdout = PIPE )
1015
1021
child_pids = []
1016
1022
if p .stdout is not None :
1017
1023
for line in p .stdout :
@@ -1020,18 +1026,16 @@ def kill_process(pid: int) -> None:
1020
1026
if local_pid .isdigit ():
1021
1027
child_pids .append (int (local_pid ))
1022
1028
try :
1023
- # Windows does not have SIGKILL, so use SIGTERM instead.
1024
- sig = getattr (signal , "SIGKILL" , signal .SIGTERM )
1025
- os .kill (pid , sig )
1029
+ os .kill (pid , signal .SIGKILL )
1026
1030
for child_pid in child_pids :
1027
1031
try :
1028
- os .kill (child_pid , sig )
1032
+ os .kill (child_pid , signal . SIGKILL )
1029
1033
except OSError :
1030
1034
pass
1031
1035
kill_check .set () # Tell the main routine that the process was killed.
1032
1036
except OSError :
1033
- # It is possible that the process gets completed in the duration after timeout
1034
- # happens and before we try to kill the process.
1037
+ # It is possible that the process gets completed in the duration after
1038
+ # timeout happens and before we try to kill the process.
1035
1039
pass
1036
1040
return
1037
1041
0 commit comments