forked from openresty/openresty-gdb-utils
-
Notifications
You must be signed in to change notification settings - Fork 0
/
isolate-worker-process
executable file
·131 lines (109 loc) · 3 KB
/
isolate-worker-process
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
#!/usr/bin/env perl
use strict;
use warnings;
use Getopt::Std qw( getopts );
use POSIX qw( SIGCONT SIGSTOP );
my %opts;
getopts("u", \%opts);
my $master = shift or
die "No master pid specified.\n";
if ($opts{u}) {
my @workers = get_child_processes($master);
for my $w (@workers) {
my $state = get_process_state($w);
if ($state eq 'T') {
warn "sending SIGCONT to the running worker $w...\n";
kill SIGCONT, $w
or die "failed to send SIGCONT to worker $w: $!";
}
}
exit;
}
my $worker = shift or
die "No worker pid specified.\n";
my $path = get_exec_path($master) or
die "Master process $master is not running or ",
"you do not have enough permissions.\n";
warn "Found master process $master: $path\n";
my @workers = get_child_processes($master);
my $found;
for my $w (@workers) {
if ($worker == $w) {
$found = 1;
last;
}
}
if (!$found) {
die "The specified worker $worker is not found in all the children of the master.\n";
}
warn "Found worker $worker under the master.\n";
my $state = get_process_state($worker);
if ($state =~ /^[ZKWPXxt]$/) {
die "Worker process $worker is in bad state: $state\n";
}
if ($state =~ /^[T]$/) {
warn "sending SIGCONT to the stopped worker $worker...\n";
kill SIGCONT, $worker
or die "failed to send SIGCONT to worker $worker: $!";
} else {
warn "the specified worker $worker is already in good state $state, no action required.\n";
}
for my $w (@workers) {
if ($worker != $w) {
my $state = get_process_state($w);
if ($state =~ /^[ZKWPXxt]$/) {
next;
}
if ($state =~ /^[RSD]$/) {
warn "sending SIGSTOP to the running worker $w...\n";
kill SIGSTOP, $w
or die "failed to send SIGSOTP to worker $w: $!";
}
}
}
sub get_process_state {
my $pid = shift;
my $infile = "/proc/$pid/stat";
open my $in, $infile
or die "Failed to open $infile for reading: $!";
my $line = <$in>;
close $in;
if (!$line) {
die "Failed to read $infile: $!";
}
chomp $line;
if ($line =~ /^\S+\s+\S+\s+(\S+)/) {
return $1;
}
die "failed to get status of process $pid: $line\n";
}
sub get_exec_path {
my $pid = shift;
my $exec_file = "/proc/$pid/exe";
if (!-f $exec_file) {
return undef;
}
return readlink $exec_file;
}
sub get_child_processes {
my $pid = shift;
my @files = glob "/proc/[0-9]*/stat";
my @children;
for my $file (@files) {
#print "file: $file\n";
if ($file =~ m{^/proc/$pid/}) {
next;
}
open my $in, $file or next;
my $line = <$in>;
close $in;
if ($line =~ /^(\d+) \S+ \S+ (\d+)/) {
my ($child, $parent) = ($1, $2);
if ($parent eq $pid) {
push @children, $child;
}
}
}
@children = sort { $a <=> $b } @children;
return @children;
}