377 lines
12 KiB
Python
377 lines
12 KiB
Python
#! py -3
|
|
"""
|
|
TODO: Make sure there's a newline at the end of the extracted files
|
|
|
|
Extract code examples from TIJ4 Refreshed. Extracts from plain text file
|
|
"""
|
|
from pathlib import Path
|
|
import sys, os
|
|
import re
|
|
import argparse
|
|
import shutil
|
|
import pprint
|
|
import difflib
|
|
from sortedcontainers import SortedSet
|
|
|
|
destination = Path('.') / "ExtractedExamples"
|
|
sourceText = Path('.') / "TIJDirectorsCut.txt"
|
|
github = Path(r'C:\Users\Bruce\Documents\GitHub\TIJ-Directors-Cut')
|
|
|
|
startBuild = """\
|
|
<?xml version="1.0" ?>
|
|
|
|
<project default="run">
|
|
<property name="chapter" value="%s"/>
|
|
<property name="excludedfiles" value="%s"/>
|
|
<import file="../Ant-Common.xml"/>
|
|
|
|
<target name="run" description="Compile and run" depends="build">
|
|
<touch file="failures"/>
|
|
"""
|
|
|
|
endBuild = """\
|
|
<delete file="failures"/>
|
|
</target>
|
|
|
|
</project>
|
|
"""
|
|
|
|
parser = argparse.ArgumentParser()
|
|
parser.add_argument("-e", "--extract", action='store_true',
|
|
help="Extract examples from TIJDirectorsCut.txt")
|
|
parser.add_argument("-x", "--clean", action='store_true',
|
|
help="Remove ExtractedExamples directory")
|
|
parser.add_argument("-c", "--compare", action='store_true',
|
|
help="Compare files from Github repository to extracted examples")
|
|
parser.add_argument("-a", "--ant", action='store_true',
|
|
help="Copy ant build files from Github repository to extracted examples")
|
|
parser.add_argument("-m", "--makeant", action='store_true',
|
|
help="Make ant files that don't exist")
|
|
|
|
|
|
def extractExamples():
|
|
if not destination.exists():
|
|
destination.mkdir()
|
|
if not sourceText.exists():
|
|
print("Cannot find", sourceText)
|
|
sys.exit()
|
|
with sourceText.open("rb") as book:
|
|
text = book.read().decode("utf-8", "ignore")
|
|
for listing in re.findall("^//:.*?///:~", text, re.DOTALL | re.MULTILINE):
|
|
title = listing.splitlines()[0]
|
|
if "//: as a special marker" in title:
|
|
continue
|
|
title = title.split()[1]
|
|
print(title)
|
|
target = destination / Path(title)
|
|
if not target.parent.exists():
|
|
target.parent.mkdir(parents=True)
|
|
with target.open("w", newline='') as codeListing:
|
|
codeListing.writelines(listing)
|
|
|
|
|
|
def clean():
|
|
print("clean")
|
|
if destination.exists():
|
|
shutil.rmtree(str(destination))
|
|
|
|
|
|
def compareWithGithub(shortForm=True):
|
|
leader = len(str(github)) + 1
|
|
githubfiles = [str(file)[leader:] for file in github.glob("**/*")]
|
|
githubfiles = [ghf for ghf in githubfiles if not ghf.startswith(".git")]
|
|
duplicates = { ghf for ghf in githubfiles if githubfiles.count(ghf) > 1 }
|
|
if duplicates:
|
|
print("duplicates = ", duplicates)
|
|
|
|
leader2 = len(str(destination)) + 1
|
|
destfiles = [str(file)[leader2:] for file in destination.glob("**/*")]
|
|
duplicates = { ghf for ghf in destfiles if destfiles.count(ghf) > 1 }
|
|
if duplicates:
|
|
print("duplicates = ", duplicates)
|
|
|
|
githubfiles = SortedSet(githubfiles)
|
|
destfiles = SortedSet(destfiles)
|
|
|
|
# print("in githubfiles but not destfiles:")
|
|
# for f in [f for f in githubfiles.difference(destfiles) if not f.endswith(".py") if not f.endswith(".xml")]:
|
|
# print("\t", f)
|
|
|
|
# print("#" * 80)
|
|
|
|
# print("in destfiles but not githubfiles:")
|
|
# for f in [f for f in destfiles.difference(githubfiles) if f.endswith(".java")]:
|
|
# # if not f.endswith(".class")
|
|
# # if not f.endswith(".xml")
|
|
# # if not f.endswith(".dat")
|
|
# # if not f.endswith(".txt")
|
|
# # if not f.endswith(".zip")
|
|
# # if not f.endswith(".out")
|
|
# # if not f.endswith(".file")
|
|
# # if not f.endswith(".gz")
|
|
# # ]:
|
|
# print("\t", f)
|
|
|
|
# print("in destfiles and in githubfiles:")
|
|
runOutput = re.compile("/\* Output:.*///:~", re.DOTALL)
|
|
differ = difflib.Differ()
|
|
|
|
def rstrip(lines):
|
|
return [line.rstrip() for line in lines]
|
|
|
|
def show(lines, sep="#"):
|
|
sys.stdout.writelines(lines)
|
|
print("\n" + sep * 80)
|
|
|
|
|
|
inBoth = [f for f in destfiles.intersection(githubfiles) if f.endswith(".java")]
|
|
for f in inBoth:
|
|
with (github / f).open() as ghf:
|
|
with (destination / f).open() as dstf:
|
|
ghblock = runOutput.sub("", ghf.read())
|
|
dstblock = runOutput.sub("", dstf.read())
|
|
if ghblock.strip() == dstblock.strip():
|
|
continue
|
|
ghtext = ghblock.splitlines(keepends=True)
|
|
#show(ghtext, sep="#")
|
|
dsttext = dstblock.splitlines(keepends=True)
|
|
#show(dsttext, sep="-")
|
|
print("[[[", f, "]]]")
|
|
if shortForm:
|
|
show([ln + "\n" for ln in difflib.context_diff(rstrip(ghtext), rstrip(dsttext))], sep="=")
|
|
else:
|
|
show([ln + "\n" for ln in differ.compare(rstrip(ghtext), rstrip(dsttext))], sep="=")
|
|
#sys.exit()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def githubDirs():
|
|
leader = len(str(github)) + 1
|
|
buildfiles = [str(file)[leader:] for file in github.glob("**/build.xml")]
|
|
return {str((github / f).parent)[leader:] for f in buildfiles}
|
|
|
|
|
|
def destDirs(pattern="**"):
|
|
leader = len(str(destination)) + 1
|
|
return {str(file)[leader:] for file in destination.glob(pattern)}
|
|
|
|
|
|
def copyAntBuildFiles():
|
|
for common in githubDirs().intersection(destDirs()):
|
|
print("->", common)
|
|
build = github / common / "build.xml"
|
|
target = destination / common
|
|
shutil.copy(str(build), str(target))
|
|
shutil.copy(str(github / "Ant-Common.xml"), str(destination))
|
|
|
|
|
|
|
|
|
|
class CodeFileOptions(object):
|
|
"""docstring for CodeFileOptions"""
|
|
def __init__(self, codeFile):
|
|
"Should probably use regular expressions for parsing instead"
|
|
self.codeFile = codeFile
|
|
self.msg = ""
|
|
|
|
self.cmdargs = None
|
|
if "{Args:" in self.codeFile.code:
|
|
for line in self.codeFile.lines:
|
|
if "{Args:" in line:
|
|
self.cmdargs = line.split("{Args:")[1].strip()
|
|
self.cmdargs = self.cmdargs.rsplit("}", 1)[0]
|
|
|
|
self.runbyhand = "{RunByHand}" in self.codeFile.code
|
|
|
|
self.exclude = None
|
|
if "{CompileTimeError}" in self.codeFile.code:
|
|
self.exclude = self.codeFile.name + ".java"
|
|
if self.codeFile.subdirs:
|
|
self.exclude = '/'.join(self.codeFile.subdirs) + '/' + self.exclude
|
|
print(self.exclude)
|
|
|
|
self.continue_on_error = "{ThrowsException}" in self.codeFile.code
|
|
|
|
self.alternatemainclass = None
|
|
if "{main: " in self.codeFile.code:
|
|
for line in self.codeFile.lines:
|
|
if "{main:" in line:
|
|
self.alternatemainclass = line.split("{main:")[1].strip()
|
|
self.alternatemainclass = self.alternatemainclass.rsplit("}", 1)[0]
|
|
|
|
self.timeout = None
|
|
if "{TimeOut:" in self.codeFile.code:
|
|
for line in self.codeFile.lines:
|
|
if "{TimeOut:" in line:
|
|
self.timeout = line.split("{TimeOut:")[1].strip()
|
|
self.timeout = self.timeout.rsplit("}", 1)[0]
|
|
self.continue_on_error = True
|
|
elif "//: gui/" in self.codeFile.code or "//: swt/" in self.codeFile.code:
|
|
self.timeout = "4000"
|
|
self.continue_on_error = True
|
|
self.msg = "* Timeout for testing *"
|
|
|
|
|
|
|
|
def classFile(self):
|
|
start = """ <jrun cls="%s" """
|
|
if self.alternatemainclass:
|
|
return start % self.alternatemainclass
|
|
if self.codeFile.package:
|
|
return start % (self.codeFile.packageName() + '.' + self.codeFile.name)
|
|
return start % self.codeFile.name
|
|
|
|
def dirPath(self):
|
|
if self.codeFile.package:
|
|
return """dirpath="%s" """ % self.codeFile.relpath
|
|
return ""
|
|
|
|
def arguments(self):
|
|
if self.cmdargs:
|
|
return """arguments='%s' """ % self.cmdargs
|
|
return ""
|
|
|
|
def failOnError(self):
|
|
if self.continue_on_error:
|
|
return """failOnError='false' """
|
|
return ""
|
|
|
|
def timeOut(self):
|
|
if self.timeout:
|
|
return """timeOut='%s' """ % self.timeout
|
|
return ""
|
|
|
|
def message(self):
|
|
if self.msg:
|
|
return """msg='%s' """ % self.msg
|
|
return ""
|
|
|
|
def createRunCommand(self):
|
|
return self.classFile() + self.dirPath() + self.arguments() + self.failOnError() + self.timeOut() + self.message() + "/>\n"
|
|
|
|
|
|
|
|
|
|
class CodeFile:
|
|
def __init__(self, javaFile, chapterDir):
|
|
self.chapter_dir = chapterDir
|
|
self.java_file = javaFile
|
|
self.subdirs = str(javaFile.parent).split("\\")[2:]
|
|
with javaFile.open() as j:
|
|
self.code = j.read()
|
|
self.lines = self.code.splitlines()
|
|
self.main = None
|
|
if "public static void main" in self.code:
|
|
self.main = True
|
|
self.package = None
|
|
if "package " in self.code:
|
|
for line in self.lines:
|
|
if line.startswith("package ") and line.strip().endswith(";"):
|
|
self.package = line
|
|
break
|
|
self.tagLine = self.lines[0][4:]
|
|
self.relpath = '../' + '/'.join(self.tagLine.split('/')[:-1])
|
|
self.name = javaFile.name.split('.')[0]
|
|
self.options = CodeFileOptions(self)
|
|
|
|
def run_command(self):
|
|
if not self.main:
|
|
return ""
|
|
return self.options.createRunCommand()
|
|
|
|
def __repr__(self):
|
|
result = self.tagLine
|
|
if self.package:
|
|
result += "\n" + self.package
|
|
result += "\n"
|
|
return result
|
|
|
|
def packageName(self):
|
|
return self.package.split()[1][:-1]
|
|
|
|
def checkPackage(self):
|
|
if not self.package:
|
|
return True
|
|
path = '.'.join(self.tagLine.split('/')[:-1])
|
|
packagePath = self.packageName()
|
|
return path == packagePath
|
|
|
|
|
|
|
|
|
|
class Chapter:
|
|
def __init__(self, dir):
|
|
self.dir = dir
|
|
self.code_files = [CodeFile(javaFile, dir) for javaFile in dir.glob("**/*.java")]
|
|
self.excludes = [cf.options.exclude for cf in self.code_files if cf.options.exclude]
|
|
|
|
def __repr__(self):
|
|
result = "-" * 80
|
|
result += "\n" + str(self.dir) + "\n"
|
|
result += "-" * 80
|
|
result += "\n"
|
|
for cf in self.code_files:
|
|
result += str(cf.name) + "\n"
|
|
return result
|
|
|
|
def checkPackages(self):
|
|
for cf in self.code_files:
|
|
if not cf.checkPackage():
|
|
print("BAD PACKAGE")
|
|
print("\t", cf.tagLine)
|
|
print("\t", cf.package)
|
|
print("\n".join(cf.lines))
|
|
|
|
def makeBuildFile(self):
|
|
buildFile = startBuild % (self.dir.name, " ".join(self.excludes))
|
|
for cf in self.code_files:
|
|
if any([cf.name + ".java" in f for f in self.excludes]) or cf.options.runbyhand:
|
|
continue
|
|
buildFile += cf.run_command()
|
|
buildFile += endBuild
|
|
with (self.dir / "build.xml").open("w") as buildxml:
|
|
buildxml.write(buildFile)
|
|
|
|
|
|
|
|
|
|
def createAntFiles():
|
|
chapters = [Chapter(fd) for fd in destination.glob("*") if fd.is_dir() if not (fd / "build.xml").exists()]
|
|
for chapter in chapters:
|
|
chapter.checkPackages()
|
|
chapter.makeBuildFile()
|
|
|
|
|
|
def default():
|
|
clean()
|
|
extractExamples()
|
|
copyAntBuildFiles()
|
|
createAntFiles()
|
|
os.chdir("ExtractedExamples")
|
|
|
|
|
|
if __name__ == '__main__':
|
|
args = parser.parse_args()
|
|
|
|
if not any(vars(args).values()): default()
|
|
|
|
if args.extract:
|
|
extractExamples()
|
|
copyAntBuildFiles()
|
|
createAntFiles()
|
|
|
|
if args.compare:
|
|
compareWithGithub()
|
|
|
|
if args.ant:
|
|
copyAntBuildFiles()
|
|
|
|
if args.makeant:
|
|
createAntFiles()
|
|
|
|
if args.clean:
|
|
clean()
|