summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--Mac/scripts/unweave.py164
1 files changed, 164 insertions, 0 deletions
diff --git a/Mac/scripts/unweave.py b/Mac/scripts/unweave.py
new file mode 100644
index 0000000..8a5e80f
--- /dev/null
+++ b/Mac/scripts/unweave.py
@@ -0,0 +1,164 @@
+"""An attempt at an unweave script.
+Jack Jansen, jack@oratrix.com, 13-Dec-00
+"""
+import re
+import sys
+import macfs
+import os
+
+BEGINDEFINITION=re.compile("^<<(?P<name>.*)>>=\s*")
+USEDEFINITION=re.compile("^(?P<pre>.*)<<(?P<name>.*)>>[^=]")
+ENDDEFINITION=re.compile("^@")
+
+class Processor:
+ def __init__(self, filename):
+ self.items = {}
+ self.filename = filename
+ self.fp = open(filename)
+ self.lineno = 0
+ self.resolving = {}
+ self.resolved = {}
+ self.pushback = None
+
+ def _readline(self):
+ """Read a line. Allow for pushback"""
+ if self.pushback:
+ rv = self.pushback
+ self.pushback = None
+ return rv
+ self.lineno = self.lineno + 1
+ return self.fp.readline()
+
+ def _linedirective(self):
+ """Return a #line cpp directive for the current input file position"""
+ return '#line %d "%s"\n'%(self.lineno-2, os.path.split(self.filename)[1])
+
+ def _readitem(self):
+ """Read the definition of an item. Insert #line where needed. """
+ rv = [self._linedirective()]
+ while 1:
+ line = self._readline()
+ if not line:
+ break
+ if ENDDEFINITION.search(line):
+ break
+ if BEGINDEFINITION.match(line):
+ self.pushback = line
+ break
+ mo = USEDEFINITION.match(line)
+ if mo:
+ pre = mo.group('pre')
+ if pre:
+ rv.append(pre+'\n')
+ rv.append(line)
+ # For simplicity we add #line directives now, if
+ # needed.
+ if mo:
+ rv.append(self._linedirective())
+ return rv
+
+ def _define(self, name, value):
+ """Define an item, or append to an existing definition"""
+ if self.items.has_key(name):
+ self.items[name] = self.items[name] + value
+ else:
+ self.items[name] = value
+
+ def read(self):
+ """Read the source file and store all definitions"""
+ while 1:
+ line = self._readline()
+ if not line: break
+ mo = BEGINDEFINITION.search(line)
+ if mo:
+ name = mo.group('name')
+ value = self._readitem()
+ self._define(name, value)
+ else:
+ pass # We could output the TeX code but we don't bother.
+
+ def resolve(self):
+ """Resolve all references"""
+ for name in self.items.keys():
+ self._resolve_one(name)
+
+ def _resolve_one(self, name):
+ """Resolve references in one definition, recursively"""
+ # First check for unknown macros and recursive calls
+ if not self.items.has_key(name):
+ print "Undefined macro:", name
+ return ['<<%s>>'%name]
+ if self.resolving.has_key(name):
+ print "Recursive macro:", name
+ return ['<<%s>>'%name]
+ # Then check that we haven't handled this one before
+ if self.resolved.has_key(name):
+ return self.items[name]
+ # No rest for the wicked: we have work to do.
+ self.resolving[name] = 1
+ result = []
+ for line in self.items[name]:
+ mo = USEDEFINITION.search(line)
+ if mo:
+ # We replace the complete line. Is this correct?
+ macro = mo.group('name')
+ replacement = self._resolve_one(macro)
+ result = result + replacement
+ else:
+ result.append(line)
+ self.items[name] = result
+ self.resolved[name] = 1
+ del self.resolving[name]
+ return result
+
+ def save(self, dir, pattern):
+ """Save macros that match pattern to folder dir"""
+ # Compile the pattern, if needed
+ if type(pattern) == type(''):
+ pattern = re.compile(pattern)
+ # If the directory is relative it is relative to the sourcefile
+ if not os.path.isabs(dir):
+ sourcedir = os.path.split(self.filename)[0]
+ dir = os.path.join(sourcedir, dir)
+ for name in self.items.keys():
+ if pattern.search(name):
+ pathname = os.path.join(dir, name)
+ data = self._stripduplines(self.items[name])
+ self._dosave(pathname, data)
+
+ def _stripduplines(self, data):
+ for i in range(len(data)-1, 0, -1):
+ if data[i][:5] == '#line' and data[i-1][:5] == '#line':
+ del data[i-1]
+ if data[-1][:5] == '#line':
+ del data[-1]
+ return data
+
+ def _dosave(self, pathname, data):
+ """Save data to pathname, unless it is identical to what is there"""
+ if os.path.exists(pathname):
+ olddata = open(pathname).readlines()
+ if olddata == data:
+ return
+ fp = open(pathname, "w").writelines(data)
+
+def process(file):
+ pr = Processor(file)
+ pr.read()
+ pr.resolve()
+ pr.save(":jacktest:src", "^.*\.cp$")
+ pr.save(":jacktest:include", "^.*\.h")
+
+def main():
+ if len(sys.argv) > 1:
+ for file in sys.argv:
+ print "Processing", file
+ process(file)
+ else:
+ fss, ok = macfs.PromptGetFile("Select .nw source file", "TEXT")
+ if not ok:
+ sys.exit(0)
+ process(fss.as_pathname())
+
+if __name__ == "__main__":
+ main()