#!/usr/local/bin/python
#
# ----------------------------------------------------------------------------
# "THE BEER-WARE LICENSE" (Revision 42, (c) Poul-Henning Kamp):
# Maxim Sobolev <sobomax@FreeBSD.org> wrote this file. As long as you retain
# this notice you can do whatever you want with this stuff. If we meet some
# day, and you think this stuff is worth it, you can buy me a beer in return.
#
# Maxim Sobolev
# ----------------------------------------------------------------------------
#
# $FreeBSD$
#
# MAINTAINER= sobomax@FreeBSD.org <- any unapproved commits to this file are
# highly discouraged!!!
#
import os, os.path, popen2, types, sys, getopt, pickle
# Global constants and semi-constants
PKG_DBDIR = '/var/db/pkg'
PORTSDIR = '/usr/ports'
ROOT_PORTMK = '/usr/share/mk/bsd.port.mk'
PLIST_FILE = '+CONTENTS'
ORIGIN_PREF = '@comment ORIGIN:'
MAKEFILE = 'Makefile'
MAKE = 'make'
ERR_PREF = 'Error:'
WARN_PREF = 'Warning:'
# Global variables
#
# PortInfo cache
picache = {}
# Useful aliases
op_isdir = os.path.isdir
op_join = os.path.join
op_split = os.path.split
op_abspath = os.path.abspath
#
# Query origin of specified installed package.
#
def getorigin(pkg):
plist = op_join(PKG_DBDIR, pkg, PLIST_FILE)
for line in open(plist).xreadlines():
if line.startswith(ORIGIN_PREF):
origin = line[len(ORIGIN_PREF):].strip()
break
else:
raise RuntimeError('%s: no origin recorded' % plist)
return origin
#
# Execute external command and return content of its stdout.
#
def getcmdout(cmdline, filterempty = 1):
pipe = popen2.Popen3(cmdline, 1)
results = pipe.fromchild.readlines()
for stream in (pipe.fromchild, pipe.tochild, pipe.childerr):
stream.close()
if pipe.wait() != 0:
if type(cmdline) is types.StringType:
cmdline = (cmdline)
raise IOError('%s: external command returned non-zero error code' % \
cmdline[0])
if filterempty != 0:
results = filter(lambda line: len(line.strip()) > 0, results)
return results
#
# For a specified path (either dir or makefile) query requested make(1)
# variables and return them as a tuple in exactly the same order as they
# were specified in function call, i.e. querymakevars('foo', 'A', 'B') will
# return a tuple with a first element being the value of A variable, and
# the second one - the value of B.
#
def querymakevars(path, *vars):
if op_isdir(path):
path = op_join(path, MAKEFILE)
dirname, makefile = op_split(path)
cmdline = [MAKE, '-f', makefile]
savedir = os.getcwd()
os.chdir(dirname)
try:
for var in vars:
cmdline.extend(('-V', var))
results = map(lambda line: line.strip(), getcmdout(cmdline, 0))
finally:
os.chdir(savedir)
return tuple(results)
def parsedeps(depstr):
return tuple(map(lambda dep: dep.split(':'), depstr.split()))
#
# For a specified port return either a new instance of the PortInfo class,
# or existing instance from the cache.
#
def getpi(path):
path = op_abspath(path)
if not picache.has_key(path):
picache[path] = PortInfo(path)
return picache[path]
#
# Format text string according to requested constrains. Useful when you have
# to display multi-line, variable width message on terminal.
#
def formatmsg(msg, wrapat = 78, seclindent = 0):
words = msg.split()
result = ''
position = 0
for word in words:
if position + 1 + len(word) > wrapat:
result += '\n' + ' ' * seclindent + word
position = seclindent + len(word)
else:
if position != 0:
result += ' '
position += 1
result += word
position += len(word)
return result
#
# Class that contain main info about the port
#
class PortInfo:
PKGNAME = None
CATEGORIES = None
MAINTAINER = None
BUILD_DEPENDS = None
LIB_DEPENDS = None
RUN_DEPENDS = None
PKGORIGIN = None
# Cached values, to speed-up things
__deps = None
__bt_deps = None
__rt_deps = None
def __init__(self, path):
self.PKGNAME, self.CATEGORIES, self.MAINTAINER, self.BUILD_DEPENDS, \
self.LIB_DEPENDS, self.RUN_DEPENDS, self.PKGORIGIN = \
querymakevars(path, 'PKGNAME', 'CATEGORIES', 'MAINTAINER', \
'BUILD_DEPENDS', 'LIB_DEPENDS', 'RUN_DEPENDS', 'PKGORIGIN')
def __str__(self):
return 'PKGNAME:\t%s\nCATEGORIES:\t%s\nMAINTAINER:\t%s\n' \
'BUILD_DEPENDS:\t%s\nLIB_DEPENDS:\t%s\nRUN_DEPENDS:\t%s\n' \
'PKGORIGIN:\t%s' % (self.PKGNAME, self.CATEGORIES, self.MAINTAINER, \
self.BUILD_DEPENDS, self.LIB_DEPENDS, self.RUN_DEPENDS, \
self.PKGORIGIN)
def getdeps(self):
if self.__deps == None:
result = []
for depstr in self.BUILD_DEPENDS, self.LIB_DEPENDS, \
self.RUN_DEPENDS:
deps = tuple(map(lambda dep: dep[1], parsedeps(depstr)))
result.append(deps)
self.__deps = tuple(result)
return self.__deps
def get_bt_deps(self):
if self.__bt_deps == None:
topdeps = self.getdeps()
topdeps = list(topdeps[0] + topdeps[1])
for dep in topdeps[:]:
botdeps = filter(lambda dep: dep not in topdeps, \
getpi(dep).get_rt_deps())
topdeps.extend(botdeps)
self.__bt_deps = tuple(topdeps)
return self.__bt_deps
def get_rt_deps(self):
if self.__rt_deps == None:
topdeps = self.getdeps()
topdeps = list(topdeps[1] + topdeps[2])
for dep in topdeps[:]:
botdeps = filter(lambda dep: dep not in topdeps, \
getpi(dep).get_rt_deps())
topdeps.extend(botdeps)
self.__rt_deps = tuple(topdeps)
return self.__rt_deps
def write_msg(*message):
if type(message) == types.StringType:
message = message,
message = tuple(filter(lambda line: line != None, message))
sys.stderr.writelines(message)
#
# Print optional message and usage information and exit with specified exit
# code.
#
def usage(code, msg = None):
myname = os.path.basename(sys.argv[0])
if msg != None:
msg = str(msg) + '\n'
write_msg(msg, "Usage: %s [-rb] [-l|L cachefile] [-s cachefile]\n" % \
myname)
sys.exit(code)
def main():
global picache
# Parse command line arguments
try:
opts, args = getopt.getopt(sys.argv[1:], 'erbl:L:s:')
except getopt.GetoptError, msg:
usage(2, msg)
if len(args) > 0 or len(opts) == 0 :
usage(2)
cachefile = None
chk_bt_deps = 0
chk_rt_deps = 0
warn_as_err = 0
for o, a in opts:
if o == '-b':
chk_bt_deps = 1
elif o == '-r':
chk_rt_deps = 1
elif o in ('-l', '-L'):
# Try to load saved PortInfo cache
try:
picache = pickle.load(open(a))
except:
picache = {}
try:
if o == '-L':
os.unlink(a)
except:
pass
elif o == '-s':
cachefile = a
elif o == '-e':
warn_as_err = 1
# Load origins of all installed packages
instpkgs = os.listdir(PKG_DBDIR)
instpkgs = filter(lambda pkg: op_isdir(op_join(PKG_DBDIR, pkg)), instpkgs)
origins = {}
for pkg in instpkgs:
origins[pkg] = getorigin(pkg)
# Resolve dependencies for the current port
info = getpi(os.getcwd())
deps = []
if chk_bt_deps != 0:
deps.extend(filter(lambda d: d not in deps, info.get_bt_deps()))
if chk_rt_deps != 0:
deps.extend(filter(lambda d: d not in deps, info.get_rt_deps()))
# Perform validation
nerrs = 0
nwarns = 0
if warn_as_err == 0:
warn_pref = WARN_PREF
else:
warn_pref = ERR_PREF
err_pref = ERR_PREF
for dep in deps:
pi = getpi(dep)
if pi.PKGORIGIN not in origins.values():
print formatmsg(seclindent = 7 * 0, msg = \
'%s package %s (%s) belongs to dependency chain, but ' \
'isn\'t installed.' % (err_pref, pi.PKGNAME, pi.PKGORIGIN))
nerrs += 1
elif pi.PKGNAME not in origins.keys():
for instpkg in origins.keys():
if origins[instpkg] == pi.PKGORIGIN:
break
print formatmsg(seclindent = 9 * 0, msg = \
'%s package %s (%s) belongs to dependency chain, but ' \
'package %s is installed instead. Perhaps it\'s an older ' \
'version - update is highly recommended.' % (warn_pref, \
pi.PKGNAME, pi.PKGORIGIN, instpkg))
nwarns += 1
# Save PortInfo cache if requested
if cachefile != None:
try:
pickle.dump(picache, open(cachefile, 'w'))
except:
pass
if warn_as_err != 0:
nerrs += nwarns
return nerrs
PORTSDIR, PKG_DBDIR = querymakevars(ROOT_PORTMK, 'PORTSDIR', 'PKG_DBDIR')
if __name__ == '__main__':
try:
sys.exit(main())
except KeyboardInterrupt:
pass