mirror of
https://github.com/python/cpython.git
synced 2024-11-24 18:34:43 +08:00
Initial revision
This commit is contained in:
parent
0481447f41
commit
ec758ead39
18
Demo/scripts/README
Normal file
18
Demo/scripts/README
Normal file
@ -0,0 +1,18 @@
|
||||
Contents of this directory:
|
||||
|
||||
byteyears.py Print product of a file's size and age
|
||||
eptags.py Create Emacs TAGS file for Python modules
|
||||
fact.py Factorize numbers
|
||||
findlinksto.py Find symbolic links to a given path (prefix)
|
||||
from.py Summarize mailbox
|
||||
lfact.py Factorize long numbers
|
||||
lpwatch.py Watch BSD line printer queues
|
||||
mkreal.py Turn a symbolic link into a real file or directory
|
||||
objgraph.py Print object graph from nm output on a library
|
||||
pdeps.py Print dependencies between Python modules
|
||||
pi.py Print digits of pi (uses arbitrary precision integers)
|
||||
primes.py Print prime numbers
|
||||
ptags.py Create vi tags file for Python modules
|
||||
suff.py Sort a list of files by suffix
|
||||
which.py Find a program in $PATH
|
||||
xxci.py Wrapper for rcsdiff and ci
|
45
Demo/scripts/fact.py
Executable file
45
Demo/scripts/fact.py
Executable file
@ -0,0 +1,45 @@
|
||||
#! /usr/local/python
|
||||
|
||||
# Factorize numbers, slowly.
|
||||
# This version uses plain integers and is thus limited to 2**31-1.
|
||||
|
||||
import sys
|
||||
from math import sqrt
|
||||
|
||||
error = 'fact.error' # exception
|
||||
|
||||
def fact(n):
|
||||
if n < 1: raise error # fact() argument should be >= 1
|
||||
if n = 1: return [] # special case
|
||||
res = []
|
||||
# Treat even factors special, so we can use i = i+2 later
|
||||
while n%2 = 0:
|
||||
res.append(2)
|
||||
n = n/2
|
||||
# Try odd numbers up to sqrt(n)
|
||||
limit = int(sqrt(float(n+1)))
|
||||
i = 3
|
||||
while i <= limit:
|
||||
if n%i = 0:
|
||||
res.append(i)
|
||||
n = n/i
|
||||
limit = int(sqrt(float(n+1)))
|
||||
else:
|
||||
i = i+2
|
||||
res.append(n)
|
||||
return res
|
||||
|
||||
def main():
|
||||
if len(sys.argv) > 1:
|
||||
for arg in sys.argv[1:]:
|
||||
n = int(eval(arg))
|
||||
print n, fact(n)
|
||||
else:
|
||||
try:
|
||||
while 1:
|
||||
n = int(input())
|
||||
print n, fact(n)
|
||||
except EOFError:
|
||||
pass
|
||||
|
||||
main()
|
25
Demo/scripts/from.py
Executable file
25
Demo/scripts/from.py
Executable file
@ -0,0 +1,25 @@
|
||||
#! /usr/local/python
|
||||
|
||||
# Print From and Subject of messages in $MAIL.
|
||||
# Extension to multiple mailboxes and other bells & whistles are left
|
||||
# as exercises for the reader.
|
||||
|
||||
import posix
|
||||
|
||||
# Open mailbox file. Exits with exception when this fails.
|
||||
|
||||
mail = open(posix.environ['MAIL'], 'r')
|
||||
|
||||
while 1:
|
||||
line = mail.readline()
|
||||
if not line: break # EOF
|
||||
if line[:5] = 'From ':
|
||||
# Start of message found
|
||||
print line[:-1],
|
||||
while 1:
|
||||
line = mail.readline()
|
||||
if not line: break # EOF
|
||||
if line = '\n': break # Blank line ends headers
|
||||
if line[:8] = 'Subject:':
|
||||
print `line[9:-1]`,
|
||||
print
|
111
Demo/scripts/lpwatch.py
Executable file
111
Demo/scripts/lpwatch.py
Executable file
@ -0,0 +1,111 @@
|
||||
#! /ufs/guido/bin/sgi/python
|
||||
#! /usr/local/python
|
||||
|
||||
# Watch line printer queue(s).
|
||||
# Intended for BSD 4.3 lpq.
|
||||
|
||||
import posix
|
||||
import sys
|
||||
import time
|
||||
import string
|
||||
|
||||
DEF_PRINTER = 'psc'
|
||||
DEF_DELAY = 10
|
||||
|
||||
def main():
|
||||
delay = DEF_DELAY # XXX Use getopt() later
|
||||
try:
|
||||
thisuser = posix.environ['LOGNAME']
|
||||
except:
|
||||
thisuser = posix.environ['USER']
|
||||
printers = sys.argv[1:]
|
||||
if not printers:
|
||||
if posix.environ.has_key('PRINTER'):
|
||||
printers = [posix.environ['PRINTER']]
|
||||
else:
|
||||
printers = [DEF_PRINTER]
|
||||
#
|
||||
clearhome = posix.popen('clear', 'r').read()
|
||||
#
|
||||
while 1:
|
||||
# Pipe output through cat for extra buffering,
|
||||
# so the output (which overwrites the previous)
|
||||
# appears instantaneous.
|
||||
sys.stdout = posix.popen('exec cat', 'w')
|
||||
sys.stdout.write(clearhome)
|
||||
for name in printers:
|
||||
pipe = posix.popen('lpq -P' + name + ' 2>&1', 'r')
|
||||
showstatus(name, pipe, thisuser)
|
||||
sts = pipe.close()
|
||||
if sts:
|
||||
print name + ': *** lpq exit status', sts
|
||||
sts = sys.stdout.close()
|
||||
time.sleep(delay)
|
||||
|
||||
def showstatus(name, pipe, thisuser):
|
||||
lines = 0
|
||||
users = {}
|
||||
aheadbytes = 0
|
||||
aheadjobs = 0
|
||||
userseen = 0
|
||||
totalbytes = 0
|
||||
totaljobs = 0
|
||||
while 1:
|
||||
line = pipe.readline()
|
||||
if not line: break
|
||||
fields = string.split(line)
|
||||
n = len(fields)
|
||||
if len(fields) >= 6 and fields[n-1] = 'bytes':
|
||||
rank = fields[0]
|
||||
user = fields[1]
|
||||
job = fields[2]
|
||||
files = fields[3:-2]
|
||||
bytes = eval(fields[n-2])
|
||||
if user = thisuser:
|
||||
userseen = 1
|
||||
elif not userseen:
|
||||
aheadbytes = aheadbytes + bytes
|
||||
aheadjobs = aheadjobs + 1
|
||||
totalbytes = totalbytes + bytes
|
||||
totaljobs = totaljobs + 1
|
||||
if users.has_key(user):
|
||||
ujobs, ubytes = users[user]
|
||||
else:
|
||||
ujobs, ubytes = 0, 0
|
||||
ujobs = ujobs + 1
|
||||
ubytes = ubytes + bytes
|
||||
users[user] = ujobs, ubytes
|
||||
else:
|
||||
if fields and fields[0] <> 'Rank':
|
||||
if line[-1:] = '\n':
|
||||
line = line[:-1]
|
||||
if not lines:
|
||||
print name + ':',
|
||||
else:
|
||||
print
|
||||
print line,
|
||||
lines = lines + 1
|
||||
if totaljobs:
|
||||
if lines > 1:
|
||||
print
|
||||
lines = lines+1
|
||||
print (totalbytes+1023)/1024, 'K',
|
||||
if totaljobs <> len(users):
|
||||
print '(' + `totaljobs` + ' jobs)',
|
||||
if len(users) = 1:
|
||||
print 'for', users.keys()[0],
|
||||
else:
|
||||
print 'for', len(users), 'users',
|
||||
if userseen:
|
||||
if aheadjobs = 0:
|
||||
print '(' + thisuser + ' first)',
|
||||
else:
|
||||
print '(' + `(aheadbytes+1023)/1024`,
|
||||
print 'K before', thisuser + ')'
|
||||
if lines:
|
||||
print
|
||||
|
||||
try:
|
||||
main()
|
||||
except KeyboardInterrupt:
|
||||
pass
|
30
Demo/scripts/pi.py
Executable file
30
Demo/scripts/pi.py
Executable file
@ -0,0 +1,30 @@
|
||||
#! /usr/local/python
|
||||
|
||||
# Print digits of pi forever.
|
||||
#
|
||||
# The algorithm, using Python's 'long' integers ("bignums"), works
|
||||
# with continued fractions, and was conceived by Lambert Meertens.
|
||||
#
|
||||
# See also the ABC Programmer's Handbook, by Geurts, Meertens & Pemberton,
|
||||
# published by Prentice-Hall (UK) Ltd., 1990.
|
||||
|
||||
import sys
|
||||
|
||||
def main():
|
||||
k, a, b, a1, b1 = 2l, 4l, 1l, 12l, 4l
|
||||
while 1:
|
||||
# Next approximation
|
||||
p, q, k = k*k, 2l*k+1l, k+1l
|
||||
a, b, a1, b1 = a1, b1, p*a+q*a1, p*b+q*b1
|
||||
# Print common digits
|
||||
d, d1 = a/b, a1/b1
|
||||
#print a, b, a1, b1
|
||||
while d = d1:
|
||||
# Use write() to avoid spaces between the digits
|
||||
sys.stdout.write(`int(d)`)
|
||||
# Flush so the output is seen immediately
|
||||
sys.stdout.flush()
|
||||
a, a1 = 10l*(a%b), 10l*(a1%b1)
|
||||
d, d1 = a/b, a1/b1
|
||||
|
||||
main()
|
26
Demo/scripts/primes.py
Executable file
26
Demo/scripts/primes.py
Executable file
@ -0,0 +1,26 @@
|
||||
#! /usr/local/python
|
||||
|
||||
# Print prime numbers in a given range
|
||||
|
||||
def main():
|
||||
import sys
|
||||
min, max = 2, 0x7fffffff
|
||||
if sys.argv[1:]:
|
||||
min = int(eval(sys.argv[1]))
|
||||
if sys.argv[2:]:
|
||||
max = int(eval(sys.argv[2]))
|
||||
primes(min, max)
|
||||
|
||||
def primes(min, max):
|
||||
if 2 >= min: print 2
|
||||
primes = [2]
|
||||
i = 3
|
||||
while i <= max:
|
||||
for p in primes:
|
||||
if i%p = 0 or p*p > i: break
|
||||
if i%p <> 0:
|
||||
primes.append(i)
|
||||
if i >= min: print i
|
||||
i = i+2
|
||||
|
||||
main()
|
29
Tools/scripts/byteyears.py
Executable file
29
Tools/scripts/byteyears.py
Executable file
@ -0,0 +1,29 @@
|
||||
#! /usr/local/python
|
||||
|
||||
# byteyears file ...
|
||||
#
|
||||
# Print a number representing the product of age and size of each file,
|
||||
# in suitable units.
|
||||
|
||||
import sys, posix, time
|
||||
from stat import *
|
||||
|
||||
secs_per_year = 365.0 * 24.0 * 3600.0
|
||||
now = time.time()
|
||||
status = 0
|
||||
|
||||
for file in sys.argv[1:]:
|
||||
try:
|
||||
st = posix.stat(file)
|
||||
except posix.error, msg:
|
||||
sys.stderr.write('can\'t stat ' + `file` + ': ' + `msg` + '\n')
|
||||
status = 1
|
||||
st = ()
|
||||
if st:
|
||||
mtime = st[ST_MTIME]
|
||||
size = st[ST_SIZE]
|
||||
age = now - mtime
|
||||
byteyears = float(size) * float(age) / secs_per_year
|
||||
print file + '\t\t' + `int(byteyears)`
|
||||
|
||||
sys.exit(status)
|
50
Tools/scripts/eptags.py
Executable file
50
Tools/scripts/eptags.py
Executable file
@ -0,0 +1,50 @@
|
||||
#! /usr/local/python
|
||||
|
||||
# eptags
|
||||
#
|
||||
# Create a TAGS file for Python programs, usable with GNU Emacs (version 18).
|
||||
# Tagged are:
|
||||
# - functions (even inside other defs or classes)
|
||||
# - classes
|
||||
# Warns about files it cannot open.
|
||||
# No warnings about duplicate tags.
|
||||
|
||||
import sys
|
||||
import regexp
|
||||
|
||||
def main():
|
||||
outfp = open('TAGS', 'w')
|
||||
args = sys.argv[1:]
|
||||
for file in args:
|
||||
treat_file(file, outfp)
|
||||
|
||||
matcher = regexp.compile('^[ \t]*(def|class)[ \t]+([a-zA-Z0-9_]+)[ \t]*\(')
|
||||
|
||||
def treat_file(file, outfp):
|
||||
try:
|
||||
fp = open(file, 'r')
|
||||
except:
|
||||
print 'Cannot open', file
|
||||
return
|
||||
charno = 0
|
||||
lineno = 0
|
||||
tags = []
|
||||
size = 0
|
||||
while 1:
|
||||
line = fp.readline()
|
||||
if not line: break
|
||||
lineno = lineno + 1
|
||||
res = matcher.exec(line)
|
||||
if res:
|
||||
(a, b), (a1, b1), (a2, b2) = res
|
||||
name = line[a2:b2]
|
||||
pat = line[a:b]
|
||||
tag = pat + '\177' + `lineno` + ',' + `charno` + '\n'
|
||||
tags.append(name, tag)
|
||||
size = size + len(tag)
|
||||
charno = charno + len(line)
|
||||
outfp.write('\f\n' + file + ',' + `size` + '\n')
|
||||
for name, tag in tags:
|
||||
outfp.write(tag)
|
||||
|
||||
main()
|
29
Tools/scripts/findlinksto.py
Executable file
29
Tools/scripts/findlinksto.py
Executable file
@ -0,0 +1,29 @@
|
||||
#! /usr/local/python
|
||||
|
||||
# findlinksto
|
||||
#
|
||||
# find symbolic links to a given path
|
||||
|
||||
import posix, path, sys
|
||||
|
||||
def visit(pattern, dirname, names):
|
||||
if path.islink(dirname):
|
||||
names[:] = []
|
||||
return
|
||||
if path.ismount(dirname):
|
||||
print 'descend into', dirname
|
||||
n = len(pattern)
|
||||
for name in names:
|
||||
name = path.cat(dirname, name)
|
||||
try:
|
||||
linkto = posix.readlink(name)
|
||||
if linkto[:n] = pattern:
|
||||
print name, '->', linkto
|
||||
except posix.error:
|
||||
pass
|
||||
|
||||
def main(pattern, args):
|
||||
for dirname in args:
|
||||
path.walk(dirname, visit, pattern)
|
||||
|
||||
main(sys.argv[1], sys.argv[2:])
|
65
Tools/scripts/mkreal.py
Executable file
65
Tools/scripts/mkreal.py
Executable file
@ -0,0 +1,65 @@
|
||||
#! /usr/local/python
|
||||
|
||||
# mkreal
|
||||
#
|
||||
# turn a symlink to a directory into a real directory
|
||||
|
||||
import sys
|
||||
import posix
|
||||
import path
|
||||
from stat import *
|
||||
|
||||
cat = path.cat
|
||||
|
||||
error = 'mkreal error'
|
||||
|
||||
BUFSIZE = 32*1024
|
||||
|
||||
def mkrealfile(name):
|
||||
st = posix.stat(name) # Get the mode
|
||||
mode = S_IMODE(st[ST_MODE])
|
||||
linkto = posix.readlink(name) # Make sure again it's a symlink
|
||||
f_in = open(name, 'r') # This ensures it's a file
|
||||
posix.unlink(name)
|
||||
f_out = open(name, 'w')
|
||||
while 1:
|
||||
buf = f_in.read(BUFSIZE)
|
||||
if not buf: break
|
||||
f_out.write(buf)
|
||||
del f_out # Flush data to disk before changing mode
|
||||
posix.chmod(name, mode)
|
||||
|
||||
def mkrealdir(name):
|
||||
st = posix.stat(name) # Get the mode
|
||||
mode = S_IMODE(st[ST_MODE])
|
||||
linkto = posix.readlink(name)
|
||||
files = posix.listdir(name)
|
||||
posix.unlink(name)
|
||||
posix.mkdir(name, mode)
|
||||
posix.chmod(name, mode)
|
||||
linkto = cat('..', linkto)
|
||||
#
|
||||
for file in files:
|
||||
if file not in ('.', '..'):
|
||||
posix.symlink(cat(linkto, file), cat(name, file))
|
||||
|
||||
def main():
|
||||
sys.stdout = sys.stderr
|
||||
progname = path.basename(sys.argv[0])
|
||||
args = sys.argv[1:]
|
||||
if not args:
|
||||
print 'usage:', progname, 'path ...'
|
||||
sys.exit(2)
|
||||
status = 0
|
||||
for name in args:
|
||||
if not path.islink(name):
|
||||
print progname+':', name+':', 'not a symlink'
|
||||
status = 1
|
||||
else:
|
||||
if path.isdir(name):
|
||||
mkrealdir(name)
|
||||
else:
|
||||
mkrealfile(name)
|
||||
sys.exit(status)
|
||||
|
||||
main()
|
213
Tools/scripts/objgraph.py
Executable file
213
Tools/scripts/objgraph.py
Executable file
@ -0,0 +1,213 @@
|
||||
#!/usr/local/python
|
||||
|
||||
# objgraph
|
||||
#
|
||||
# Read "nm -o" input (on IRIX: "nm -Bo") of a set of libraries or modules
|
||||
# and print various interesting listings, such as:
|
||||
#
|
||||
# - which names are used but not defined in the set (and used where),
|
||||
# - which names are defined in the set (and where),
|
||||
# - which modules use which other modules,
|
||||
# - which modules are used by which other modules.
|
||||
#
|
||||
# Usage: objgraph [-cdu] [file] ...
|
||||
# -c: print callers per objectfile
|
||||
# -d: print callees per objectfile
|
||||
# -u: print usage of undefined symbols
|
||||
# If none of -cdu is specified, all are assumed.
|
||||
# Use "nm -o" to generate the input (on IRIX: "nm -Bo"),
|
||||
# e.g.: nm -o /lib/libc.a | objgraph
|
||||
|
||||
|
||||
import sys
|
||||
import string
|
||||
import path
|
||||
import getopt
|
||||
import regexp
|
||||
|
||||
# Types of symbols.
|
||||
#
|
||||
definitions = 'TRGDSBAEC'
|
||||
externals = 'UV'
|
||||
ignore = 'Nntrgdsbavuc'
|
||||
|
||||
# Regular expression to parse "nm -o" output.
|
||||
#
|
||||
matcher = regexp.compile('(.*):\t?........ (.) (.*)$')
|
||||
|
||||
# Store "item" in "dict" under "key".
|
||||
# The dictionary maps keys to lists of items.
|
||||
# If there is no list for the key yet, it is created.
|
||||
#
|
||||
def store(dict, key, item):
|
||||
if dict.has_key(key):
|
||||
dict[key].append(item)
|
||||
else:
|
||||
dict[key] = [item]
|
||||
|
||||
# Return a flattened version of a list of strings: the concatenation
|
||||
# of its elements with intervening spaces.
|
||||
#
|
||||
def flat(list):
|
||||
s = ''
|
||||
for item in list:
|
||||
s = s + ' ' + item
|
||||
return s[1:]
|
||||
|
||||
# Global variables mapping defined/undefined names to files and back.
|
||||
#
|
||||
file2undef = {}
|
||||
def2file = {}
|
||||
file2def = {}
|
||||
undef2file = {}
|
||||
|
||||
# Read one input file and merge the data into the tables.
|
||||
# Argument is an open file.
|
||||
#
|
||||
def readinput(file):
|
||||
while 1:
|
||||
s = file.readline(200) # Arbitrary, but reasonable limit
|
||||
if not s:
|
||||
break
|
||||
# If you get an exception on this line,
|
||||
# it is probably caused by an unexpected input line:
|
||||
(ra, rb), (r1a, r1b), (r2a, r2b), (r3a, r3b) = matcher.exec(s)
|
||||
fn, name, type = s[r1a:r1b], s[r3a:r3b], s[r2a:r2b]
|
||||
if type in definitions:
|
||||
store(def2file, name, fn)
|
||||
store(file2def, fn, name)
|
||||
elif type in externals:
|
||||
store(file2undef, fn, name)
|
||||
store(undef2file, name, fn)
|
||||
elif not type in ignore:
|
||||
print fn + ':' + name + ': unknown type ' + type
|
||||
|
||||
# Print all names that were undefined in some module and where they are
|
||||
# defined.
|
||||
#
|
||||
def printcallee():
|
||||
flist = file2undef.keys()
|
||||
flist.sort()
|
||||
for file in flist:
|
||||
print file + ':'
|
||||
elist = file2undef[file]
|
||||
elist.sort()
|
||||
for ext in elist:
|
||||
if len(ext) >= 8:
|
||||
tabs = '\t'
|
||||
else:
|
||||
tabs = '\t\t'
|
||||
if not def2file.has_key(ext):
|
||||
print '\t' + ext + tabs + ' *undefined'
|
||||
else:
|
||||
print '\t' + ext + tabs + flat(def2file[ext])
|
||||
|
||||
# Print for each module the names of the other modules that use it.
|
||||
#
|
||||
def printcaller():
|
||||
files = file2def.keys()
|
||||
files.sort()
|
||||
for file in files:
|
||||
callers = []
|
||||
for label in file2def[file]:
|
||||
if undef2file.has_key(label):
|
||||
callers = callers + undef2file[label]
|
||||
if callers:
|
||||
callers.sort()
|
||||
print file + ':'
|
||||
lastfn = ''
|
||||
for fn in callers:
|
||||
if fn <> lastfn:
|
||||
print '\t' + fn
|
||||
lastfn = fn
|
||||
else:
|
||||
print file + ': unused'
|
||||
|
||||
# Print undefine names and where they are used.
|
||||
#
|
||||
def printundef():
|
||||
undefs = {}
|
||||
for file in file2undef.keys():
|
||||
for ext in file2undef[file]:
|
||||
if not def2file.has_key(ext):
|
||||
store(undefs, ext, file)
|
||||
elist = undefs.keys()
|
||||
elist.sort()
|
||||
for ext in elist:
|
||||
print ext + ':'
|
||||
flist = undefs[ext]
|
||||
flist.sort()
|
||||
for file in flist:
|
||||
print '\t' + file
|
||||
|
||||
# Print warning messages about names defined in more than one file.
|
||||
#
|
||||
def warndups():
|
||||
savestdout = sys.stdout
|
||||
sys.stdout = sys.stderr
|
||||
names = def2file.keys()
|
||||
names.sort()
|
||||
for name in names:
|
||||
if len(def2file[name]) > 1:
|
||||
print 'warning:', name, 'multiply defined:',
|
||||
print flat(def2file[name])
|
||||
sys.stdout = savestdout
|
||||
|
||||
# Main program
|
||||
#
|
||||
def main():
|
||||
try:
|
||||
optlist, args = getopt.getopt(sys.argv[1:], 'cdu')
|
||||
except getopt.error:
|
||||
sys.stdout = sys.stderr
|
||||
print 'Usage:', path.basename(sys.argv[0]), '[-cdu] [file] ...'
|
||||
print '-c: print callers per objectfile'
|
||||
print '-d: print callees per objectfile'
|
||||
print '-u: print usage of undefined symbols'
|
||||
print 'If none of -cdu is specified, all are assumed.'
|
||||
print 'Use "nm -o" to generate the input (on IRIX: "nm -Bo"),'
|
||||
print 'e.g.: nm -o /lib/libc.a | objgraph'
|
||||
return 1
|
||||
optu = optc = optd = 0
|
||||
for opt, void in optlist:
|
||||
if opt = '-u':
|
||||
optu = 1
|
||||
elif opt = '-c':
|
||||
optc = 1
|
||||
elif opt = '-d':
|
||||
optd = 1
|
||||
if optu = optc = optd = 0:
|
||||
optu = optc = optd = 1
|
||||
if not args:
|
||||
args = ['-']
|
||||
for file in args:
|
||||
if file = '-':
|
||||
readinput(sys.stdin)
|
||||
else:
|
||||
readinput(open(file, 'r'))
|
||||
#
|
||||
warndups()
|
||||
#
|
||||
more = (optu + optc + optd > 1)
|
||||
if optd:
|
||||
if more:
|
||||
print '---------------All callees------------------'
|
||||
printcallee()
|
||||
if optu:
|
||||
if more:
|
||||
print '---------------Undefined callees------------'
|
||||
printundef()
|
||||
if optc:
|
||||
if more:
|
||||
print '---------------All Callers------------------'
|
||||
printcaller()
|
||||
return 0
|
||||
|
||||
# Call the main program.
|
||||
# Use its return value as exit status.
|
||||
# Catch interrupts to avoid stack trace.
|
||||
#
|
||||
try:
|
||||
sys.exit(main())
|
||||
except KeyboardInterrupt:
|
||||
sys.exit(1)
|
167
Tools/scripts/pdeps.py
Executable file
167
Tools/scripts/pdeps.py
Executable file
@ -0,0 +1,167 @@
|
||||
#! /usr/local/python
|
||||
|
||||
# pdeps
|
||||
#
|
||||
# Find dependencies between a bunch of Python modules.
|
||||
#
|
||||
# Usage:
|
||||
# pdeps file1.py file2.py ...
|
||||
#
|
||||
# Output:
|
||||
# Four tables separated by lines like '--- Closure ---':
|
||||
# 1) Direct dependencies, listing which module imports which other modules
|
||||
# 2) The inverse of (1)
|
||||
# 3) Indirect dependencies, or the closure of the above
|
||||
# 4) The inverse of (3)
|
||||
#
|
||||
# To do:
|
||||
# - command line options to select output type
|
||||
# - option to automatically scan the Python library for referenced modules
|
||||
# - option to limit output to particular modules
|
||||
|
||||
|
||||
import sys
|
||||
import regexp
|
||||
import path
|
||||
import string
|
||||
|
||||
|
||||
# Main program
|
||||
#
|
||||
def main():
|
||||
args = sys.argv[1:]
|
||||
if not args:
|
||||
print 'usage: pdeps file.py file.py ...'
|
||||
return 2
|
||||
#
|
||||
table = {}
|
||||
for arg in args:
|
||||
process(arg, table)
|
||||
#
|
||||
print '--- Uses ---'
|
||||
printresults(table)
|
||||
#
|
||||
print '--- Used By ---'
|
||||
inv = inverse(table)
|
||||
printresults(inv)
|
||||
#
|
||||
print '--- Closure of Uses ---'
|
||||
reach = closure(table)
|
||||
printresults(reach)
|
||||
#
|
||||
print '--- Closure of Used By ---'
|
||||
invreach = inverse(reach)
|
||||
printresults(invreach)
|
||||
#
|
||||
return 0
|
||||
|
||||
|
||||
# Compiled regular expressions to search for import statements
|
||||
#
|
||||
m_import = regexp.compile('^[ \t]*from[ \t]+([^ \t]+)[ \t]+')
|
||||
m_from = regexp.compile('^[ \t]*import[ \t]+([^#]+)')
|
||||
|
||||
|
||||
# Collect data from one file
|
||||
#
|
||||
def process(filename, table):
|
||||
fp = open(filename, 'r')
|
||||
mod = path.basename(filename)
|
||||
if mod[-3:] = '.py':
|
||||
mod = mod[:-3]
|
||||
table[mod] = list = []
|
||||
while 1:
|
||||
line = fp.readline()
|
||||
if not line: break
|
||||
while line[-1:] = '\\':
|
||||
nextline = fp.readline()
|
||||
if not nextline: break
|
||||
line = line[:-1] + nextline
|
||||
result = m_import.exec(line)
|
||||
if not result:
|
||||
result = m_from.exec(line)
|
||||
if result:
|
||||
(a, b), (a1, b1) = result
|
||||
words = string.splitfields(line[a1:b1], ',')
|
||||
# print '#', line, words
|
||||
for word in words:
|
||||
word = string.strip(word)
|
||||
if word not in list:
|
||||
list.append(word)
|
||||
|
||||
|
||||
# Compute closure (this is in fact totally general)
|
||||
#
|
||||
def closure(table):
|
||||
modules = table.keys()
|
||||
#
|
||||
# Initialize reach with a copy of table
|
||||
#
|
||||
reach = {}
|
||||
for mod in modules:
|
||||
reach[mod] = table[mod][:]
|
||||
#
|
||||
# Iterate until no more change
|
||||
#
|
||||
change = 1
|
||||
while change:
|
||||
change = 0
|
||||
for mod in modules:
|
||||
for mo in reach[mod]:
|
||||
if mo in modules:
|
||||
for m in reach[mo]:
|
||||
if m not in reach[mod]:
|
||||
reach[mod].append(m)
|
||||
change = 1
|
||||
#
|
||||
return reach
|
||||
|
||||
|
||||
# Invert a table (this is again totally general).
|
||||
# All keys of the original table are made keys of the inverse,
|
||||
# so there may be empty lists in the inverse.
|
||||
#
|
||||
def inverse(table):
|
||||
inv = {}
|
||||
for key in table.keys():
|
||||
if not inv.has_key(key):
|
||||
inv[key] = []
|
||||
for item in table[key]:
|
||||
store(inv, item, key)
|
||||
return inv
|
||||
|
||||
|
||||
# Store "item" in "dict" under "key".
|
||||
# The dictionary maps keys to lists of items.
|
||||
# If there is no list for the key yet, it is created.
|
||||
#
|
||||
def store(dict, key, item):
|
||||
if dict.has_key(key):
|
||||
dict[key].append(item)
|
||||
else:
|
||||
dict[key] = [item]
|
||||
|
||||
|
||||
# Tabulate results neatly
|
||||
#
|
||||
def printresults(table):
|
||||
modules = table.keys()
|
||||
maxlen = 0
|
||||
for mod in modules: maxlen = max(maxlen, len(mod))
|
||||
modules.sort()
|
||||
for mod in modules:
|
||||
list = table[mod]
|
||||
list.sort()
|
||||
print string.ljust(mod, maxlen), ':',
|
||||
if mod in list:
|
||||
print '(*)',
|
||||
for ref in list:
|
||||
print ref,
|
||||
print
|
||||
|
||||
|
||||
# Call main and honor exit status
|
||||
try:
|
||||
sys.exit(main())
|
||||
except KeyboardInterrupt:
|
||||
sys.exit(1)
|
49
Tools/scripts/ptags.py
Executable file
49
Tools/scripts/ptags.py
Executable file
@ -0,0 +1,49 @@
|
||||
#! /usr/local/python
|
||||
|
||||
# ptags
|
||||
#
|
||||
# Create a tags file for Python programs, usable with vi.
|
||||
# Tagged are:
|
||||
# - functions (even inside other defs or classes)
|
||||
# - classes
|
||||
# - filenames
|
||||
# Warns about files it cannot open.
|
||||
# No warnings about duplicate tags.
|
||||
|
||||
import sys
|
||||
import regexp
|
||||
import path
|
||||
|
||||
tags = [] # Modified global variable!
|
||||
|
||||
def main():
|
||||
args = sys.argv[1:]
|
||||
for file in args: treat_file(file)
|
||||
if tags:
|
||||
fp = open('tags', 'w')
|
||||
tags.sort()
|
||||
for s in tags: fp.write(s)
|
||||
|
||||
matcher = regexp.compile('^[ \t]*(def|class)[ \t]+([a-zA-Z0-9_]+)[ \t]*\(')
|
||||
|
||||
def treat_file(file):
|
||||
try:
|
||||
fp = open(file, 'r')
|
||||
except:
|
||||
print 'Cannot open', file
|
||||
return
|
||||
base = path.basename(file)
|
||||
if base[-3:] = '.py': base = base[:-3]
|
||||
s = base + '\t' + file + '\t' + '1\n'
|
||||
tags.append(s)
|
||||
while 1:
|
||||
line = fp.readline()
|
||||
if not line: break
|
||||
res = matcher.exec(line)
|
||||
if res:
|
||||
(a, b), (a1, b1), (a2, b2) = res
|
||||
name = line[a2:b2]
|
||||
s = name + '\t' + file + '\t/^' + line[a:b] + '/\n'
|
||||
tags.append(s)
|
||||
|
||||
main()
|
29
Tools/scripts/suff.py
Executable file
29
Tools/scripts/suff.py
Executable file
@ -0,0 +1,29 @@
|
||||
#! /usr/local/python
|
||||
|
||||
# suff
|
||||
#
|
||||
# show different suffixes amongst arguments
|
||||
|
||||
import sys
|
||||
|
||||
def main():
|
||||
files = sys.argv[1:]
|
||||
suffixes = {}
|
||||
for file in files:
|
||||
suff = getsuffix(file)
|
||||
if not suffixes.has_key(suff):
|
||||
suffixes[suff] = []
|
||||
suffixes[suff].append(file)
|
||||
keys = suffixes.keys()
|
||||
keys.sort()
|
||||
for suff in keys:
|
||||
print `suff`, len(suffixes[suff])
|
||||
|
||||
def getsuffix(file):
|
||||
suff = ''
|
||||
for i in range(len(file)):
|
||||
if file[i] = '.':
|
||||
suff = file[i:]
|
||||
return suff
|
||||
|
||||
main()
|
44
Tools/scripts/which.py
Executable file
44
Tools/scripts/which.py
Executable file
@ -0,0 +1,44 @@
|
||||
#! /usr/local/python
|
||||
|
||||
# Variant of "which".
|
||||
# On stderr, near and total misses are reported.
|
||||
|
||||
import sys, posix, string, path
|
||||
from stat import *
|
||||
|
||||
def msg(str):
|
||||
sys.stderr.write(str + '\n')
|
||||
|
||||
pathlist = string.splitfields(posix.environ['PATH'], ':')
|
||||
|
||||
sts = 0
|
||||
|
||||
for prog in sys.argv[1:]:
|
||||
ident = ()
|
||||
for dir in pathlist:
|
||||
file = path.cat(dir, prog)
|
||||
try:
|
||||
st = posix.stat(file)
|
||||
if S_ISREG(st[ST_MODE]):
|
||||
mode = S_IMODE(st[ST_MODE])
|
||||
if mode % 2 or mode/8 % 2 or mode/64 % 2:
|
||||
if ident:
|
||||
if st[:3] = ident:
|
||||
s = ': same as '
|
||||
else:
|
||||
s = ': also '
|
||||
msg(prog + s + file)
|
||||
else:
|
||||
print file
|
||||
ident = st[:3]
|
||||
else:
|
||||
msg(file + ': not executable')
|
||||
else:
|
||||
msg(file + ': not a disk file')
|
||||
except posix.error:
|
||||
pass
|
||||
if not ident:
|
||||
msg(prog + ': not found')
|
||||
sts = 1
|
||||
|
||||
sys.exit(sts)
|
77
Tools/scripts/xxci.py
Executable file
77
Tools/scripts/xxci.py
Executable file
@ -0,0 +1,77 @@
|
||||
#! /usr/local/python
|
||||
|
||||
# xxci
|
||||
#
|
||||
# check in files for which rcsdiff returns nonzero exit status
|
||||
|
||||
import sys
|
||||
import posix
|
||||
import stat
|
||||
import path
|
||||
import commands
|
||||
|
||||
MAXSIZE = 200*1024 # Files this big must be binaries and are skipped.
|
||||
|
||||
def getargs():
|
||||
args = sys.argv[1:]
|
||||
if args:
|
||||
return args
|
||||
print 'No arguments, checking almost *'
|
||||
for file in posix.listdir('.'):
|
||||
if not skipfile(file):
|
||||
args.append(file)
|
||||
if not args:
|
||||
print 'Nothing to do -- exit 1'
|
||||
sys.exit(1)
|
||||
args.sort()
|
||||
return args
|
||||
|
||||
badnames = ['tags', 'xyzzy']
|
||||
badprefixes = ['.', ',', '@', '#', 'o.']
|
||||
badsuffixes = \
|
||||
['~', '.a', '.o', '.old', '.bak', '.orig', '.new', '.prev', '.not']
|
||||
# XXX Should generalize even more to use fnmatch!
|
||||
|
||||
def skipfile(file):
|
||||
if file in badnames or \
|
||||
badprefix(file) or badsuffix(file) or \
|
||||
path.islink(file) or path.isdir(file):
|
||||
return 1
|
||||
# Skip huge files -- probably binaries.
|
||||
try:
|
||||
st = posix.stat(file)
|
||||
except posix.error:
|
||||
return 1 # Doesn't exist -- skip it
|
||||
return st[stat.ST_SIZE] >= MAXSIZE
|
||||
|
||||
def badprefix(file):
|
||||
for bad in badprefixes:
|
||||
if file[:len(bad)] = bad: return 1
|
||||
return 0
|
||||
|
||||
def badsuffix(file):
|
||||
for bad in badsuffixes:
|
||||
if file[-len(bad):] = bad: return 1
|
||||
return 0
|
||||
|
||||
def go(args):
|
||||
for file in args:
|
||||
print file + ':'
|
||||
if run('rcsdiff -c', file):
|
||||
if askyesno('Check in ' + file + ' ? '):
|
||||
sts = run('rcs -l', file) # ignored
|
||||
# can't use run() here because it's interactive
|
||||
sts = posix.system('ci -l ' + file)
|
||||
|
||||
def run(cmd, file):
|
||||
sts, output = commands.getstatusoutput(cmd + commands.mkarg(file))
|
||||
if sts:
|
||||
print output
|
||||
print 'Exit status', sts
|
||||
return sts
|
||||
|
||||
def askyesno(prompt):
|
||||
s = raw_input(prompt)
|
||||
return s in ['y', 'yes']
|
||||
|
||||
go(getargs())
|
Loading…
Reference in New Issue
Block a user