-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathutils.py
314 lines (257 loc) · 8.76 KB
/
utils.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
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
#!/usr/bin/python
#
# Copyright (c) 2003 Intel Corporation
# All rights reserved.
#
# Copyright (c) 2004-2006 The Trustees of Princeton University
# All rights reserved.
# expected /proc/partitions format
import os, sys, shutil
import subprocess
import shlex
import socket
import fcntl
import string
import exceptions
from Exceptions import *
####################
# the simplest way to debug is to let the node take off,
# ssh into it as root using the debug ssh key in /etc/planetlab
# then go to /tmp/source
# edit this file locally to turn on breakpoints if needed, then run
# ./BootManager.py
####################
### handling breakpoints in the startup process
import select, sys, string
### global debugging settings
# enabling this will cause the node to ask for breakpoint-mode at startup
# production code should read False/False
PROMPT_MODE=False
# default for when prompt is turned off, or it's on but the timeout triggers
BREAKPOINT_MODE=False
# verbose mode is just fine
VERBOSE_MODE=True
# in seconds : if no input, proceed
PROMPT_TIMEOUT=5
def prompt_for_breakpoint_mode ():
global BREAKPOINT_MODE
if PROMPT_MODE:
default_answer=BREAKPOINT_MODE
answer=''
if BREAKPOINT_MODE:
display="[y]/n"
else:
display="y/[n]"
sys.stdout.write ("Want to run in breakpoint mode ? %s "%display)
sys.stdout.flush()
r,w,e = select.select ([sys.stdin],[],[],PROMPT_TIMEOUT)
if r:
answer = string.strip(sys.stdin.readline())
else:
sys.stdout.write("\nTimed-out (%d s)"%PROMPT_TIMEOUT)
if answer:
BREAKPOINT_MODE = ( answer == "y" or answer == "Y")
else:
BREAKPOINT_MODE = default_answer
label="Off"
if BREAKPOINT_MODE:
label="On"
sys.stdout.write("\nCurrent BREAKPOINT_MODE is %s\n"%label)
def breakpoint (message, cmd = None):
if BREAKPOINT_MODE:
if cmd is None:
cmd="/bin/sh"
message=message+" -- Entering bash - type ^D to proceed"
print message
os.system(cmd)
########################################
def makedirs( path ):
"""
from python docs for os.makedirs:
Throws an error exception if the leaf directory
already exists or cannot be created.
That is real useful. Instead, we'll create the directory, then use a
separate function to test for its existance.
Return 1 if the directory exists and/or has been created, a BootManagerException
otherwise. Does not test the writability of said directory.
"""
try:
os.makedirs( path )
except OSError:
pass
try:
os.listdir( path )
except OSError:
raise BootManagerException, "Unable to create directory tree: %s" % path
return 1
def removedir( path ):
"""
remove a directory tree, return 1 if successful, a BootManagerException
if failure.
"""
try:
os.listdir( path )
except OSError:
return 1
try:
shutil.rmtree( path )
except OSError, desc:
raise BootManagerException, "Unable to remove directory tree: %s" % path
return 1
def sysexec( cmd, log=None, fsck=False, shell=False ):
"""
execute a system command, output the results to the logger
if log <> None
return 1 if command completed (return code of non-zero),
0 if failed. A BootManagerException is raised if the command
was unable to execute or was interrupted by the user with Ctrl+C
"""
try:
# Thierry - Jan. 6 2011
# would probably make sense to look for | here as well
# however this is fragile and hard to test thoroughly
# let the caller set 'shell' when that is desirable
if shell or cmd.__contains__(">"):
prog = subprocess.Popen(cmd, shell=True)
if log is not None:
log.write("sysexec (shell mode) >>> %s" % cmd)
if VERBOSE_MODE:
print "sysexec (shell mode) >>> %s" % cmd
else:
prog = subprocess.Popen(shlex.split(cmd), stdout=subprocess.PIPE, stderr=subprocess.PIPE)
if log is not None:
log.write("sysexec >>> %s\n" % cmd)
if VERBOSE_MODE:
print "sysexec >>> %s" % cmd
except OSError:
raise BootManagerException, \
"Unable to create instance of subprocess.Popen " \
"for command: %s" % cmd
try:
(stdoutdata, stderrdata) = prog.communicate()
except KeyboardInterrupt:
raise BootManagerException, "Interrupted by user"
# log stdout & stderr
if log is not None:
if stdoutdata:
log.write("==========stdout\n"+stdoutdata)
if stderrdata:
log.write("==========stderr\n"+stderrdata)
returncode = prog.wait()
if fsck:
# The exit code returned by fsck is the sum of the following conditions:
# 0 - No errors
# 1 - File system errors corrected
# 2 - System should be rebooted
# 4 - File system errors left uncorrected
# 8 - Operational error
# 16 - Usage or syntax error
# 32 - Fsck canceled by user request
# 128 - Shared library error
if returncode != 0 and returncode != 1:
raise BootManagerException, "Running %s failed (rc=%d)" % (cmd,returncode)
else:
if returncode != 0:
raise BootManagerException, "Running %s failed (rc=%d)" % (cmd,returncode)
prog = None
return 1
def sysexec_chroot( path, cmd, log=None, shell=False):
"""
same as sysexec, but inside a chroot
"""
preload = ""
release = os.uname()[2]
# 2.6.12 kernels need this
if release[:5] == "2.6.1":
library = "%s/lib/libc-opendir-hack.so" % path
if not os.path.exists(library):
shutil.copy("./libc-opendir-hack.so", library)
preload = "/bin/env LD_PRELOAD=/lib/libc-opendir-hack.so"
sysexec("chroot %s %s %s" % (path, preload, cmd), log, shell=shell)
def sysexec_chroot_noerr( path, cmd, log=None, shell=False ):
"""
same as sysexec_chroot, but capture boot manager exceptions
"""
try:
rc= 0
rc= sysexec_chroot( cmd, log, shell=shell )
except BootManagerException, e:
pass
return rc
def sysexec_noerr( cmd, log=None, shell=False ):
"""
same as sysexec, but capture boot manager exceptions
"""
try:
rc= 0
rc= sysexec( cmd, log, shell=shell )
except BootManagerException, e:
pass
return rc
def chdir( dir ):
"""
change to a directory, return 1 if successful, a BootManagerException if failure
"""
try:
os.chdir( dir )
except OSError:
raise BootManagerException, "Unable to change to directory: %s" % dir
return 1
def removefile( filepath ):
"""
removes a file, return 1 if successful, 0 if failure
"""
try:
os.remove( filepath )
except OSError:
raise BootManagerException, "Unable to remove file: %s" % filepath
return 1
# from: http://forums.devshed.com/archive/t-51149/
# Ethernet-card-address-Through-Python-or-C
def hexy(n):
return "%02x" % (ord(n))
def get_mac_from_interface(ifname):
"""
given a device name, like eth0, return its mac_address.
return None if the device doesn't exist.
"""
SIOCGIFHWADDR = 0x8927 # magic number
s=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
ifname = string.strip(ifname)
ifr = ifname + '\0'*(32-len(ifname))
try:
r= fcntl.ioctl(s.fileno(),SIOCGIFHWADDR,ifr)
addr = map(hexy,r[18:24])
ret = (':'.join(map(str, addr)))
except IOError, e:
ret = None
return ret
def check_file_hash(filename, hash_filename):
"""Check the file's integrity with a given hash."""
return sha1_file(filename) == open(hash_filename).read().split()[0].strip()
def sha1_file(filename):
"""Calculate sha1 hash of file."""
try:
try:
import hashlib
m = hashlib.sha1()
except:
import sha
m=sha.new()
f = file(filename, 'rb')
while True:
# 256 KB seems ideal for speed/memory tradeoff
# It wont get much faster with bigger blocks, but
# heap peak grows
block = f.read(256 * 1024)
if len(block) == 0:
# end of file
break
m.update(block)
# Simple trick to keep total heap even lower
# Delete the previous block, so while next one is read
# we wont have two allocated blocks with same size
del block
return m.hexdigest()
except IOError:
raise BootManagerException, "Cannot calculate SHA1 hash of %s" % filename