OnJava8-Examples/tools/Examples.py

367 lines
11 KiB
Python
Raw Normal View History

2015-04-27 14:31:25 -07:00
#! py -3
"""
TODO: Make sure there's a newline at the end of the extracted files
2015-04-27 14:31:25 -07:00
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
2015-04-27 14:31:25 -07:00
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"/>
2015-04-27 21:27:42 -07:00
<target name="run" description="Compile and run" depends="build">
2015-04-27 14:31:25 -07:00
<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):
2015-04-27 14:31:25 -07:00
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 }
2015-04-27 21:27:42 -07:00
if duplicates:
print("duplicates = ", duplicates)
2015-04-27 14:31:25 -07:00
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 }
2015-04-27 21:27:42 -07:00
if duplicates:
print("duplicates = ", duplicates)
2015-04-27 14:31:25 -07:00
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()
2015-04-27 14:31:25 -07:00
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))
2015-04-28 14:17:58 -07:00
2015-04-28 10:54:50 -07:00
class CodeFileOptions(object):
"""docstring for CodeFileOptions"""
def __init__(self, codeFile):
2015-04-28 14:17:58 -07:00
"Should probably use regular expressions for parsing instead"
2015-04-28 10:54:50 -07:00
self.codeFile = codeFile
self.cmdargs = None
if "{Args:" in self.codeFile.code:
for line in self.codeFile.lines:
if "{Args:" in line:
2015-04-28 14:17:58 -07:00
self.cmdargs = line.split("{Args:")[1].strip()
self.cmdargs = self.cmdargs.rsplit("}", 1)[0]
2015-04-28 10:54:50 -07:00
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
2015-04-28 10:54:50 -07:00
2015-04-28 14:17:58 -07:00
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
2015-04-28 14:17:58 -07:00
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:
2015-04-28 14:17:58 -07:00
return """failOnError='false' """
return ""
def timeOut(self):
if self.timeout:
return """timeOut='%s' """ % self.timeout
return ""
2015-04-28 14:17:58 -07:00
def createRunCommand(self):
return self.classFile() + self.dirPath() + self.arguments() + self.failOnError() + self.timeOut() + "/>\n"
2015-04-28 14:17:58 -07:00
2015-04-27 14:31:25 -07:00
class CodeFile:
2015-04-27 21:27:42 -07:00
def __init__(self, javaFile, chapterDir):
self.chapter_dir = chapterDir
self.java_file = javaFile
self.subdirs = str(javaFile.parent).split("\\")[2:]
2015-04-27 14:31:25 -07:00
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]
2015-04-28 10:54:50 -07:00
self.options = CodeFileOptions(self)
2015-04-28 00:00:38 -07:00
def run_command(self):
if not self.main:
return ""
2015-04-28 14:17:58 -07:00
return self.options.createRunCommand()
2015-04-27 14:31:25 -07:00
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
2015-04-28 14:17:58 -07:00
2015-04-27 14:31:25 -07:00
class Chapter:
def __init__(self, dir):
self.dir = dir
2015-04-27 21:27:42 -07:00
self.code_files = [CodeFile(javaFile, dir) for javaFile in dir.glob("**/*.java")]
2015-04-28 10:54:50 -07:00
self.excludes = [cf.options.exclude for cf in self.code_files if cf.options.exclude]
2015-04-27 14:31:25 -07:00
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:
2015-04-28 10:54:50 -07:00
if any([cf.name + ".java" in f for f in self.excludes]) or cf.options.runbyhand:
2015-04-27 14:31:25 -07:00
continue
2015-04-28 00:00:38 -07:00
buildFile += cf.run_command()
2015-04-27 14:31:25 -07:00
buildFile += endBuild
with (self.dir / "build.xml").open("w") as buildxml:
buildxml.write(buildFile)
2015-04-28 14:17:58 -07:00
2015-04-27 14:31:25 -07:00
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()