#!/usr/bin/python # -*- Mode: Python -*- # vi:si:et:sw=4:sts=4:ts=4 """ parse, update and write .signals and .args files """ from twisted.python import util import sys import os def debug(*args): pass class Object: def __init__(self, name): self._signals = util.OrderedDict() self._args = util.OrderedDict() self.name = name def __repr__(self): return "" % self.name def add_signal(self, signal, overwrite=True): if not overwrite and self._signals.has_key(signal.name): raise IndexError, "signal %s already in %r" % (signal.name, self) self._signals[signal.name] = signal def add_arg(self, arg, overwrite=True): if not overwrite and self._args.has_key(arg.name): raise IndexError, "arg %s already in %r" % (arg.name, self) self._args[arg.name] = arg class Docable: def __init__(self, **kwargs): for key in self.attrs: setattr(self, key, kwargs[key]) self.dict = kwargs def __repr__(self): return "<%r %s>" % (str(self.__class__), self.name) class Signal(Docable): attrs = ['name', 'returns', 'args'] class Arg(Docable): attrs = ['name', 'type', 'range', 'flags', 'nick', 'blurb', 'default'] class GDoc: def load_file(self, filename): try: lines = open(filename).readlines() self.load_data("".join(lines)) except IOError: print "WARNING - could not read from %s" % filename def save_file(self, filename, backup=False): """ Save the signals information to the given .signals file if the file content changed. """ olddata = None try: lines = open(filename).readlines() olddata = "".join(lines) except IOError: print "WARNING - could not read from %s" % filename newdata = self.get_data() if olddata and olddata == newdata: return if olddata: if backup: os.rename(filename, filename + '.bak') handle = open(filename, "w") handle.write(newdata) handle.close() class Signals(GDoc): def __init__(self): self._objects = util.OrderedDict() def load_data(self, data): """ Load the .signals lines, creating our list of objects and signals. """ import re smatcher = re.compile( '(?s)' # make . match \n '\n(.*?)\n' ) nmatcher = re.compile( '' '(?P\S*)' # store object '::' '(?P\S*)' # store signal '' ) rmatcher = re.compile( '(?s)' # make . match \n '(?P\S*)\n' # store returns '(?P.*)' # store args ) for block in smatcher.findall(data): nmatch = nmatcher.search(block) if nmatch: o = nmatch.group('object') debug("Found object", o) debug("Found signal", nmatch.group('signal')) if not self._objects.has_key(o): object = Object(o) self._objects[o] = object rmatch = rmatcher.search(block) if rmatch: dict = rmatch.groupdict().copy() dict['name'] = nmatch.group('signal') signal = Signal(**dict) self._objects[o].add_signal(signal) def get_data(self): lines = [] for o in self._objects.values(): for s in o._signals.values(): block = """ %(object)s::%(name)s %(returns)s %(args)s """ d = s.dict.copy() d['object'] = o.name lines.append(block % d) return "\n".join(lines) + '\n' class Args(GDoc): def __init__(self): self._objects = util.OrderedDict() def load_data(self, data): """ Load the .args lines, creating our list of objects and args. """ import re amatcher = re.compile( '(?s)' # make . match \n '\n(.*?)\n' ) nmatcher = re.compile( '' '(?P\S*)' # store object '::' '(?P\S*)' # store arg '' ) rmatcher = re.compile( '(?s)' # make . match \n '(?P\S*)\n' # store type '(?P.*?)\n' # store range '(?P\S*)\n' # store flags '(?P.*?)\n' # store nick '(?P.*?)\n' # store blurb '(?P.*?)\n' # store default ) for block in amatcher.findall(data): nmatch = nmatcher.search(block) if nmatch: o = nmatch.group('object') debug("Found object", o) debug("Found arg", nmatch.group('arg')) if not self._objects.has_key(o): object = Object(o) self._objects[o] = object rmatch = rmatcher.search(block) if rmatch: dict = rmatch.groupdict().copy() dict['name'] = nmatch.group('arg') arg = Arg(**dict) self._objects[o].add_arg(arg) else: print "ERROR: could not match arg from block %s" % block def get_data(self): lines = [] for o in self._objects.values(): for a in o._args.values(): block = """ %(object)s::%(name)s %(type)s %(range)s %(flags)s %(nick)s %(blurb)s %(default)s """ d = a.dict.copy() d['object'] = o.name lines.append(block % d) return "\n".join(lines) + '\n' def main(argv): modulename = None try: modulename = argv[1] except IndexError: sys.stderr.write('Pleae provide a documentation module name\n') sys.exit(1) print "Merging scangobj output for %s" % modulename signals = Signals() signals.load_file(modulename + '.signals') signals.load_file(modulename + '.signals.new') signals.save_file(modulename + '.signals', backup=True) args = Args() args.load_file(modulename + '.args') args.load_file(modulename + '.args.new') args.save_file(modulename + '.args', backup=True) main(sys.argv)