forked from etotheipi/BitcoinArmory
-
Notifications
You must be signed in to change notification settings - Fork 174
/
guardian.py
113 lines (87 loc) · 3.27 KB
/
guardian.py
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
from __future__ import print_function
################################################################################
# #
# Copyright (C) 2011-2015, Armory Technologies, Inc. #
# Distributed under the GNU Affero General Public License (AGPL v3) #
# See LICENSE or http://www.gnu.org/licenses/agpl.html #
# #
################################################################################
import sys
import psutil
import time
import platform
import os
import signal
import subprocess
# Note: this ended up not being used for Windows! Check out guardian.exe
opsys = platform.system()
OS_WINDOWS = 'win32' in opsys.lower() or 'windows' in opsys.lower()
try:
pid_armory = int(sys.argv[1])
pid_bitcoind = int(sys.argv[2])
except:
print('USAGE: %d armorypid bitcoindpid' % sys.argv[0])
exit(0)
def check_pid(pid, name=''):
try:
proc = psutil.Process(pid)
procstr = ' '.join(proc.cmdline)
if name=='':
return procstr
else:
return procstr if procstr==name else False
except psutil.error.NoSuchProcess:
return False
def kill(pid):
if OS_WINDOWS:
#import ctypes
#k32 = ctypes.windll.kernel32
#handle = k32.OpenProcess(1,0,pid)
#return (0 != k32.TerminateProcess(handle,0))
os.kill(pid, signal.CTRL_C_EVENT)
else:
os.kill(pid, signal.SIGTERM)
time.sleep(3)
if not check_pid(pid):
return
print('Regular TERMINATE of bitcoind failed; issuing SIGKILL (hard)')
time.sleep(1)
os.kill(pid, signal.SIGKILL)
################################################################################
def killProcessTree(pid):
# In this case, Windows is easier because we know it has the get_children
# call, because have bundled a recent version of psutil. Linux, however,
# does not have that function call in earlier versions.
if OS_WINDOWS:
for child in psutil.Process(pid).get_children():
kill(child.pid)
else:
proc = subprocess.Popen("ps -o pid --ppid %d --noheaders" % pid, shell=True, stdout=subprocess.PIPE)
out,err = proc.communicate()
for pid_str in out.split("\n")[:-1]:
kill(int(pid_str))
# Verify the two PIDs are valid
proc_name_armory = check_pid(pid_armory)
proc_name_bitcoind = check_pid(pid_bitcoind)
if proc_name_armory:
print('ArmoryQt is running in pid=%d (%s)' % (pid_armory, proc_name_armory))
else:
print('ArmoryQt IS NOT RUNNING!')
if proc_name_bitcoind:
print('bitcoind is running in pid=%d (%s)' % (pid_bitcoind, proc_name_bitcoind))
else:
print('bitcoind IS NOT RUNNING!')
while True:
time.sleep(3)
if not check_pid(pid_armory, proc_name_armory):
#print 'ArmoryQt died!'
break
if not check_pid(pid_bitcoind, proc_name_bitcoind):
#print 'bitcoind disappeared -- guardian exiting'
exit(0)
if check_pid(pid_bitcoind, proc_name_bitcoind):
# Depending on how popen was called, bitcoind may be a child of
# pid_bitcoind. But psutil makes it easy to find those child procs
# and kill them.
killProcessTree(pid_bitcoind)
kill(pid_bitcoind)