2015-09-11 15:34:01 +02:00
|
|
|
"""Helpers for writing unit tests."""
|
|
|
|
|
|
|
|
from __future__ import print_function, division, absolute_import
|
2016-12-21 13:20:20 +00:00
|
|
|
from __future__ import unicode_literals
|
|
|
|
import collections
|
2015-09-11 15:34:01 +02:00
|
|
|
from fontTools.misc.py23 import *
|
2015-09-30 20:19:15 +01:00
|
|
|
from fontTools.misc.xmlWriter import XMLWriter
|
2015-09-11 15:34:01 +02:00
|
|
|
|
|
|
|
|
|
|
|
def parseXML(xmlSnippet):
|
|
|
|
"""Parses a snippet of XML.
|
|
|
|
|
2016-12-21 13:20:20 +00:00
|
|
|
Input can be either a single string (unicode or UTF-8 bytes), or a
|
|
|
|
a sequence of strings.
|
|
|
|
|
2015-09-11 15:34:01 +02:00
|
|
|
The result is in the same format that would be returned by
|
|
|
|
XMLReader, but the parser imposes no constraints on the root
|
|
|
|
element so it can be called on small snippets of TTX files.
|
|
|
|
"""
|
|
|
|
# To support snippets with multiple elements, we add a fake root.
|
|
|
|
reader = TestXMLReader_()
|
2016-12-21 14:12:43 +00:00
|
|
|
xml = b"<root>"
|
2016-12-21 13:20:20 +00:00
|
|
|
if isinstance(xmlSnippet, bytes):
|
2016-12-21 14:12:43 +00:00
|
|
|
xml += xmlSnippet
|
2016-12-21 13:20:20 +00:00
|
|
|
elif isinstance(xmlSnippet, unicode):
|
2016-12-21 14:12:43 +00:00
|
|
|
xml += tobytes(xmlSnippet, 'utf-8')
|
2016-12-21 13:20:20 +00:00
|
|
|
elif isinstance(xmlSnippet, collections.Iterable):
|
2016-12-21 14:12:43 +00:00
|
|
|
xml += b"".join(tobytes(s, 'utf-8') for s in xmlSnippet)
|
2016-12-21 13:20:20 +00:00
|
|
|
else:
|
|
|
|
raise TypeError("expected string or sequence of strings; found %r"
|
|
|
|
% type(xmlSnippet).__name__)
|
2016-12-21 14:12:43 +00:00
|
|
|
xml += b"</root>"
|
2016-12-21 13:20:20 +00:00
|
|
|
reader.parser.Parse(xml, 0)
|
2015-09-11 15:34:01 +02:00
|
|
|
return reader.root[2]
|
|
|
|
|
|
|
|
|
2015-10-17 06:47:52 +02:00
|
|
|
class FakeFont:
|
|
|
|
def __init__(self, glyphs):
|
|
|
|
self.glyphOrder_ = glyphs
|
2016-10-14 17:13:56 +02:00
|
|
|
self.lazy = False
|
2017-01-10 17:55:30 +01:00
|
|
|
self.tables = {}
|
|
|
|
|
|
|
|
def __getitem__(self, tag):
|
|
|
|
return self.tables[tag]
|
2015-10-17 06:47:52 +02:00
|
|
|
|
2017-04-18 19:18:46 +02:00
|
|
|
def __setitem__(self, tag, table):
|
|
|
|
self.tables[tag] = table
|
|
|
|
|
|
|
|
def get(self, tag, default=None):
|
|
|
|
return self.tables.get(tag, default)
|
|
|
|
|
2015-10-17 06:47:52 +02:00
|
|
|
def getGlyphID(self, name):
|
|
|
|
return self.glyphOrder_.index(name)
|
|
|
|
|
|
|
|
def getGlyphName(self, glyphID):
|
|
|
|
if glyphID < len(self.glyphOrder_):
|
|
|
|
return self.glyphOrder_[glyphID]
|
|
|
|
else:
|
|
|
|
return "glyph%.5d" % glyphID
|
|
|
|
|
|
|
|
def getGlyphOrder(self):
|
|
|
|
return self.glyphOrder_
|
|
|
|
|
|
|
|
|
2015-09-11 15:34:01 +02:00
|
|
|
class TestXMLReader_(object):
|
|
|
|
def __init__(self):
|
|
|
|
from xml.parsers.expat import ParserCreate
|
|
|
|
self.parser = ParserCreate()
|
|
|
|
self.parser.StartElementHandler = self.startElement_
|
|
|
|
self.parser.EndElementHandler = self.endElement_
|
|
|
|
self.parser.CharacterDataHandler = self.addCharacterData_
|
|
|
|
self.root = None
|
|
|
|
self.stack = []
|
|
|
|
|
|
|
|
def startElement_(self, name, attrs):
|
|
|
|
element = (name, attrs, [])
|
|
|
|
if self.stack:
|
|
|
|
self.stack[-1][2].append(element)
|
|
|
|
else:
|
|
|
|
self.root = element
|
|
|
|
self.stack.append(element)
|
|
|
|
|
|
|
|
def endElement_(self, name):
|
|
|
|
self.stack.pop()
|
|
|
|
|
|
|
|
def addCharacterData_(self, data):
|
|
|
|
self.stack[-1][2].append(data)
|
2015-09-30 20:19:15 +01:00
|
|
|
|
|
|
|
|
2016-12-21 13:20:20 +00:00
|
|
|
def makeXMLWriter(newlinestr='\n'):
|
2015-09-30 20:19:15 +01:00
|
|
|
# don't write OS-specific new lines
|
2016-12-21 13:20:20 +00:00
|
|
|
writer = XMLWriter(BytesIO(), newlinestr=newlinestr)
|
2015-09-30 20:19:15 +01:00
|
|
|
# erase XML declaration
|
|
|
|
writer.file.seek(0)
|
|
|
|
writer.file.truncate()
|
2016-10-14 21:04:35 +02:00
|
|
|
return writer
|
|
|
|
|
|
|
|
|
|
|
|
def getXML(func, ttFont=None):
|
2016-12-21 13:20:20 +00:00
|
|
|
"""Call the passed toXML function and return the written content as a
|
|
|
|
list of lines (unicode strings).
|
2016-10-14 21:04:35 +02:00
|
|
|
Result is stripped of XML declaration and OS-specific newline characters.
|
|
|
|
"""
|
|
|
|
writer = makeXMLWriter()
|
2016-01-14 16:25:28 +01:00
|
|
|
func(writer, ttFont)
|
2015-09-30 20:19:15 +01:00
|
|
|
xml = writer.file.getvalue().decode("utf-8")
|
2016-12-21 13:20:20 +00:00
|
|
|
# toXML methods must always end with a writer.newline()
|
|
|
|
assert xml.endswith("\n")
|
|
|
|
return xml.splitlines()
|
2016-12-26 14:34:29 -05:00
|
|
|
|
|
|
|
|
|
|
|
class MockFont(object):
|
|
|
|
"""A font-like object that automatically adds any looked up glyphname
|
|
|
|
to its glyphOrder."""
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
self._glyphOrder = ['.notdef']
|
|
|
|
class AllocatingDict(dict):
|
|
|
|
def __missing__(reverseDict, key):
|
|
|
|
self._glyphOrder.append(key)
|
|
|
|
gid = len(reverseDict)
|
|
|
|
reverseDict[key] = gid
|
|
|
|
return gid
|
|
|
|
self._reverseGlyphOrder = AllocatingDict({'.notdef': 0})
|
|
|
|
self.lazy = False
|
|
|
|
|
|
|
|
def getGlyphID(self, glyph, requireReal=None):
|
|
|
|
gid = self._reverseGlyphOrder[glyph]
|
|
|
|
return gid
|
|
|
|
|
|
|
|
def getReverseGlyphMap(self):
|
|
|
|
return self._reverseGlyphOrder
|
|
|
|
|
|
|
|
def getGlyphName(self, gid):
|
|
|
|
return self._glyphOrder[gid]
|
|
|
|
|
|
|
|
def getGlyphOrder(self):
|
|
|
|
return self._glyphOrder
|