gecko-dev/layout/reftests/w3c-css/import-tests.py

384 строки
13 KiB
Python
Executable File

#!/usr/bin/python
# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
import os
from optparse import OptionParser
from subprocess import Popen, PIPE
import xml.dom.minidom
import html5lib
import fnmatch
import shutil
import sys
import re
# FIXME:
# * Import more tests rather than just the very limited set currently
# chosen.
# * Read in a (checked-in) input file with a list of test assertions
# expected to fail.
# * Read in a (checked-in) input file with a list of reference choices
# for tests with multiple rel="match" references. (But still go
# though all those references in case they, in turn, have references.)
# Eventually we should import all the tests that have references. (At
# least for a subset of secs. And we probably want to organize the
# directory structure by spec to avoid constant file moves when files
# move in the W3C repository. And we probably also want to import each
# test only once, even if it covers more than one spec.)
# But for now, let's just import a few sets of tests.
gSubtrees = [
os.path.join("css-namespaces"),
os.path.join("css-conditional"),
os.path.join("css-values"),
os.path.join("css-multicol"),
os.path.join("css-writing-modes"),
os.path.join("selectors"),
]
gPrefixedProperties = [
"column-count",
"column-fill",
"column-gap",
"column-rule",
"column-rule-color",
"column-rule-style",
"column-rule-width",
"columns",
"column-span",
"column-width"
]
gPrefixRegexp = re.compile(
r"([^-#]|^)(" + r"|".join(gPrefixedProperties) + r")\b")
# Map of about:config prefs that need toggling, for a given test subdirectory.
# Entries should look like:
# "$SUBDIR_NAME": "pref($PREF_NAME, $PREF_VALUE)"
#
# For example, when "@supports" was behind a pref, gDefaultPreferences had:
# "css3-conditional": "pref(layout.css.supports-rule.enabled,true)"
gDefaultPreferences = {
}
gLog = None
gFailList = []
gDestPath = None
gSrcPath = None
support_dirs_mapped = set()
filemap = {}
speclinkmap = {}
propsaddedfor = []
tests = []
gOptions = None
gArgs = None
gTestfiles = []
gTestFlags = {}
def to_unix_path_sep(path):
return path.replace('\\', '/')
def log_output_of(subprocess):
global gLog
subprocess.wait()
if (subprocess.returncode != 0):
raise StandardError("error while running subprocess")
gLog.write(subprocess.stdout.readline().rstrip())
def write_log_header():
global gLog, gSrcPath
gLog.write("Importing revision: ")
log_output_of(Popen(["git", "rev-parse", "HEAD"],
stdout=PIPE, cwd=gSrcPath))
gLog.write("\nfrom repository: ")
branches = Popen(["git", "branch", "--format",
"%(HEAD)%(upstream:lstrip=2)"],
stdout=PIPE, cwd=gSrcPath)
for branch in branches.stdout:
if branch[0] == "*":
upstream = branch[1:].split("/")[0]
break
if len(upstream.strip()) == 0:
raise StandardError("No upstream repository found")
log_output_of(Popen(["git", "remote", "get-url", upstream],
stdout=PIPE, cwd=gSrcPath))
gLog.write("\n")
def remove_existing_dirs():
global gDestPath
# Remove existing directories that we're going to regenerate. This
# is necessary so that we can give errors in cases where our import
# might copy two files to the same location, which we do by giving
# errors if a copy would overwrite a file.
for dirname in os.listdir(gDestPath):
fulldir = os.path.join(gDestPath, dirname)
if not os.path.isdir(fulldir):
continue
shutil.rmtree(fulldir)
def populate_test_files():
global gSubtrees, gTestfiles
excludeDirs = ["support", "reftest", "reference", "reports", "tools"]
for subtree in gSubtrees:
for dirpath, dirnames, filenames in os.walk(subtree, topdown=True):
for exclDir in excludeDirs:
if exclDir in dirnames:
dirnames.remove(exclDir)
for f in filenames:
if f == "README" or \
f.find("-ref.") != -1:
continue
gTestfiles.append(os.path.join(dirpath, f))
gTestfiles.sort()
def copy_file(test, srcfile, destname, isSupportFile=False):
global gDestPath, gLog, gSrcPath
if not srcfile.startswith(gSrcPath):
raise StandardError("Filename " + srcfile + " does not start with " + gSrcPath)
logname = srcfile[len(gSrcPath):]
gLog.write("Importing " + to_unix_path_sep(logname) +
" to " + to_unix_path_sep(destname) + "\n")
destfile = os.path.join(gDestPath, destname)
destdir = os.path.dirname(destfile)
if not os.path.exists(destdir):
os.makedirs(destdir)
if os.path.exists(destfile):
raise StandardError("file " + destfile + " already exists")
copy_and_prefix(test, srcfile, destfile, isSupportFile)
def copy_support_files(test, dirname):
global gSrcPath
if dirname in support_dirs_mapped:
return
support_dirs_mapped.add(dirname)
support_dir = os.path.join(dirname, "support")
if not os.path.exists(support_dir):
return
for dirpath, dirnames, filenames in os.walk(support_dir):
for srcname in filenames:
if srcname == "LOCK":
continue
full_srcname = os.path.join(dirpath, srcname)
destname = to_unix_path_sep(os.path.relpath(full_srcname, gSrcPath))
copy_file(test, full_srcname, destname, True)
def map_file(srcname):
global gSrcPath
srcname = to_unix_path_sep(os.path.normpath(srcname))
if srcname in filemap:
return filemap[srcname]
destname = to_unix_path_sep(os.path.relpath(srcname, gSrcPath))
destdir = os.path.dirname(destname)
filemap[srcname] = destname
load_flags_for(srcname, destname)
copy_file(destname, srcname, destname, False)
copy_support_files(destname, os.path.dirname(srcname))
return destname
def load_flags_for(srcname, destname):
global gTestFlags
gTestFlags[destname] = []
if not (is_html(srcname) or is_xml(srcname)):
return
document = get_document_for(srcname)
for meta in document.getElementsByTagName("meta"):
name = meta.getAttribute("name")
if name == "flags":
gTestFlags[destname] = meta.getAttribute("content").split()
def is_html(fn):
return fn.endswith(".htm") or fn.endswith(".html")
def is_xml(fn):
return fn.endswith(".xht") or fn.endswith(".xml") or fn.endswith(".xhtml") or fn.endswith(".svg")
def get_document_for(srcname):
document = None # an xml.dom.minidom document
if is_html(srcname):
# An HTML file
f = open(srcname, "rb")
parser = html5lib.HTMLParser(tree=html5lib.treebuilders.getTreeBuilder("dom"))
document = parser.parse(f)
f.close()
else:
# An XML file
document = xml.dom.minidom.parse(srcname)
return document
def add_test_items(srcname):
if not (is_html(srcname) or is_xml(srcname)):
map_file(srcname)
return None
document = get_document_for(srcname)
refs = []
notrefs = []
for link in document.getElementsByTagName("link"):
rel = link.getAttribute("rel")
if rel == "match":
arr = refs
elif rel == "mismatch":
arr = notrefs
else:
continue
if str(link.getAttribute("href")) != "":
arr.append(os.path.join(os.path.dirname(srcname), str(link.getAttribute("href"))))
else:
gLog.write("Warning: href attribute found empty in " + srcname + "\n")
if len(refs) > 1:
raise StandardError("Need to add code to specify which reference we want to match.")
for ref in refs:
tests.append(["==", map_file(srcname), map_file(ref)])
for notref in notrefs:
tests.append(["!=", map_file(srcname), map_file(notref)])
# Add chained references too
for ref in refs:
add_test_items(ref)
for notref in notrefs:
add_test_items(notref)
AHEM_FONT_PATH = os.path.normpath(
os.path.join(os.path.dirname(__file__), "../fonts/Ahem.ttf"))
AHEM_DECL_CONTENT = """@font-face {{
font-family: Ahem;
src: url("{}");
}}"""
AHEM_DECL_HTML = """<style type="text/css">
""" + AHEM_DECL_CONTENT + """
</style>
"""
AHEM_DECL_XML = """<style type="text/css"><![CDATA[
""" + AHEM_DECL_CONTENT + """
]]></style>
"""
def copy_and_prefix(test, aSourceFileName, aDestFileName, isSupportFile=False):
global gTestFlags, gPrefixRegexp
newFile = open(aDestFileName, 'wb')
unPrefixedFile = open(aSourceFileName, 'rb')
testName = aDestFileName[len(gDestPath)+1:]
ahemFontAdded = False
for line in unPrefixedFile:
replacementLine = line
searchRegex = "\s*<style\s*"
if not isSupportFile and not ahemFontAdded and 'ahem' in gTestFlags[test] and re.search(searchRegex, line):
# First put our ahem font declation before the first <style>
# element
template = AHEM_DECL_HTML if is_html(aDestFileName) else AHEM_DECL_XML
ahemPath = os.path.relpath(AHEM_FONT_PATH, os.path.dirname(aDestFileName))
newFile.write(template.format(to_unix_path_sep(ahemPath)))
ahemFontAdded = True
replacementLine = gPrefixRegexp.sub(r"\1-moz-\2", replacementLine)
newFile.write(replacementLine)
newFile.close()
unPrefixedFile.close()
def read_options():
global gArgs, gOptions
op = OptionParser()
op.usage = \
'''%prog <clone of git repository>
Import CSS reftests from a web-platform-tests git repository clone.
The location of the git repository must be given on the command
line.'''
(gOptions, gArgs) = op.parse_args()
if len(gArgs) != 1:
op.error("Too few arguments specified.")
def setup_paths():
global gSubtrees, gDestPath, gSrcPath
# FIXME: generate gSrcPath with consistent trailing / regardless of input.
# (We currently expect the argument to have a trailing slash.)
gSrcPath = gArgs[0]
if not os.path.isdir(gSrcPath) or \
not os.path.isdir(os.path.join(gSrcPath, ".git")):
raise StandardError("source path does not appear to be a git clone")
gSrcPath = os.path.join(gSrcPath, "css") + "/"
if not os.path.isdir(gSrcPath):
raise StandardError("source path does not appear to be " +
"a wpt clone which contains css tests")
gDestPath = os.path.join(os.path.dirname(os.path.realpath(__file__)), "received")
newSubtrees = []
for relPath in gSubtrees:
newSubtrees[len(gSubtrees):] = [os.path.join(gSrcPath, relPath)]
gSubtrees = newSubtrees
def setup_log():
global gLog
# Since we're going to commit the tests, we should also commit
# information about where they came from.
gLog = open(os.path.join(gDestPath, "import.log"), "wb")
def read_fail_list():
global gFailList
dirname = os.path.realpath(__file__).split(os.path.sep)
dirname = os.path.sep.join(dirname[:len(dirname)-1])
with open(os.path.join(dirname, "failures.list"), "rb") as f:
for line in f:
line = line.strip()
if not line or line.startswith("#"):
continue
items = line.split()
refpat = None
if items[-1].startswith("ref:"):
refpat = re.compile(fnmatch.translate(items.pop()[4:]))
pat = re.compile(fnmatch.translate(items.pop()))
gFailList.append((pat, refpat, items))
def main():
global gDestPath, gLog, gTestfiles, gTestFlags, gFailList
read_options()
setup_paths()
read_fail_list()
setup_log()
write_log_header()
remove_existing_dirs()
populate_test_files()
for t in gTestfiles:
add_test_items(t)
listfile = open(os.path.join(gDestPath, "reftest.list"), "wb")
listfile.write("# THIS FILE IS AUTOGENERATED BY {0}\n# DO NOT EDIT!\n".format(os.path.basename(__file__)))
lastDefaultPreferences = None
for test in tests:
defaultPreferences = gDefaultPreferences.get(test[1].split("/")[0], None)
if defaultPreferences != lastDefaultPreferences:
if defaultPreferences is None:
listfile.write("\ndefault-preferences\n\n")
else:
listfile.write("\ndefault-preferences {0}\n\n".format(defaultPreferences))
lastDefaultPreferences = defaultPreferences
key = 1
while not test[key] in gTestFlags.keys() and key < len(test):
key = key + 1
testType = test[key - 1]
testFlags = gTestFlags[test[key]]
# Replace the Windows separators if any. Our internal strings
# all use the system separator, however the failure/skip lists
# and reftest.list always use '/' so we fix the paths here.
test[key] = to_unix_path_sep(test[key])
test[key + 1] = to_unix_path_sep(test[key + 1])
testKey = test[key]
refKey = test[key + 1]
fail = []
for pattern, refpattern, failureType in gFailList:
if (refpattern is None or refpattern.match(refKey)) and \
pattern.match(testKey):
fail = failureType
test = fail + test
listfile.write(" ".join(test) + "\n")
listfile.close()
gLog.close()
if __name__ == '__main__':
main()