| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197 |
- """Various low level data validators."""
- from __future__ import annotations
- import calendar
- from collections.abc import Mapping, Sequence
- from io import open
- import fontTools.misc.filesystem as fs
- from typing import Any, Type, Optional, Union
- from fontTools.annotations import IntFloat
- from fontTools.ufoLib.utils import numberTypes
- GenericDict = dict[str, tuple[Union[type, tuple[Type[Any], ...]], bool]]
- # -------
- # Generic
- # -------
- def isDictEnough(value: Any) -> bool:
- """
- Some objects will likely come in that aren't
- dicts but are dict-ish enough.
- """
- if isinstance(value, Mapping):
- return True
- for attr in ("keys", "values", "items"):
- if not hasattr(value, attr):
- return False
- return True
- def genericTypeValidator(value: Any, typ: Type[Any]) -> bool:
- """
- Generic. (Added at version 2.)
- """
- return isinstance(value, typ)
- def genericIntListValidator(values: Any, validValues: Sequence[int]) -> bool:
- """
- Generic. (Added at version 2.)
- """
- if not isinstance(values, (list, tuple)):
- return False
- valuesSet = set(values)
- validValuesSet = set(validValues)
- if valuesSet - validValuesSet:
- return False
- for value in values:
- if not isinstance(value, int):
- return False
- return True
- def genericNonNegativeIntValidator(value: Any) -> bool:
- """
- Generic. (Added at version 3.)
- """
- if not isinstance(value, int):
- return False
- if value < 0:
- return False
- return True
- def genericNonNegativeNumberValidator(value: Any) -> bool:
- """
- Generic. (Added at version 3.)
- """
- if not isinstance(value, numberTypes):
- return False
- if value < 0:
- return False
- return True
- def genericDictValidator(value: Any, prototype: GenericDict) -> bool:
- """
- Generic. (Added at version 3.)
- """
- # not a dict
- if not isinstance(value, Mapping):
- return False
- # missing required keys
- for key, (typ, required) in prototype.items():
- if not required:
- continue
- if key not in value:
- return False
- # unknown keys
- for key in value.keys():
- if key not in prototype:
- return False
- # incorrect types
- for key, v in value.items():
- prototypeType, required = prototype[key]
- if v is None and not required:
- continue
- if not isinstance(v, prototypeType):
- return False
- return True
- # --------------
- # fontinfo.plist
- # --------------
- # Data Validators
- def fontInfoStyleMapStyleNameValidator(value: Any) -> bool:
- """
- Version 2+.
- """
- options = ["regular", "italic", "bold", "bold italic"]
- return value in options
- def fontInfoOpenTypeGaspRangeRecordsValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- if not isinstance(value, list):
- return False
- if len(value) == 0:
- return True
- validBehaviors = [0, 1, 2, 3]
- dictPrototype: GenericDict = dict(
- rangeMaxPPEM=(int, True), rangeGaspBehavior=(list, True)
- )
- ppemOrder = []
- for rangeRecord in value:
- if not genericDictValidator(rangeRecord, dictPrototype):
- return False
- ppem = rangeRecord["rangeMaxPPEM"]
- behavior = rangeRecord["rangeGaspBehavior"]
- ppemValidity = genericNonNegativeIntValidator(ppem)
- if not ppemValidity:
- return False
- behaviorValidity = genericIntListValidator(behavior, validBehaviors)
- if not behaviorValidity:
- return False
- ppemOrder.append(ppem)
- if ppemOrder != sorted(ppemOrder):
- return False
- return True
- def fontInfoOpenTypeHeadCreatedValidator(value: Any) -> bool:
- """
- Version 2+.
- """
- # format: 0000/00/00 00:00:00
- if not isinstance(value, str):
- return False
- # basic formatting
- if not len(value) == 19:
- return False
- if value.count(" ") != 1:
- return False
- strDate, strTime = value.split(" ")
- if strDate.count("/") != 2:
- return False
- if strTime.count(":") != 2:
- return False
- # date
- strYear, strMonth, strDay = strDate.split("/")
- if len(strYear) != 4:
- return False
- if len(strMonth) != 2:
- return False
- if len(strDay) != 2:
- return False
- try:
- intYear = int(strYear)
- intMonth = int(strMonth)
- intDay = int(strDay)
- except ValueError:
- return False
- if intMonth < 1 or intMonth > 12:
- return False
- monthMaxDay = calendar.monthrange(intYear, intMonth)[1]
- if intDay < 1 or intDay > monthMaxDay:
- return False
- # time
- strHour, strMinute, strSecond = strTime.split(":")
- if len(strHour) != 2:
- return False
- if len(strMinute) != 2:
- return False
- if len(strSecond) != 2:
- return False
- try:
- intHour = int(strHour)
- intMinute = int(strMinute)
- intSecond = int(strSecond)
- except ValueError:
- return False
- if intHour < 0 or intHour > 23:
- return False
- if intMinute < 0 or intMinute > 59:
- return False
- if intSecond < 0 or intSecond > 59:
- return False
- # fallback
- return True
- def fontInfoOpenTypeNameRecordsValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- if not isinstance(value, list):
- return False
- dictPrototype: GenericDict = dict(
- nameID=(int, True),
- platformID=(int, True),
- encodingID=(int, True),
- languageID=(int, True),
- string=(str, True),
- )
- for nameRecord in value:
- if not genericDictValidator(nameRecord, dictPrototype):
- return False
- return True
- def fontInfoOpenTypeOS2WeightClassValidator(value: Any) -> bool:
- """
- Version 2+.
- """
- if not isinstance(value, int):
- return False
- if value < 0:
- return False
- return True
- def fontInfoOpenTypeOS2WidthClassValidator(value: Any) -> bool:
- """
- Version 2+.
- """
- if not isinstance(value, int):
- return False
- if value < 1:
- return False
- if value > 9:
- return False
- return True
- def fontInfoVersion2OpenTypeOS2PanoseValidator(values: Any) -> bool:
- """
- Version 2.
- """
- if not isinstance(values, (list, tuple)):
- return False
- if len(values) != 10:
- return False
- for value in values:
- if not isinstance(value, int):
- return False
- # XXX further validation?
- return True
- def fontInfoVersion3OpenTypeOS2PanoseValidator(values: Any) -> bool:
- """
- Version 3+.
- """
- if not isinstance(values, (list, tuple)):
- return False
- if len(values) != 10:
- return False
- for value in values:
- if not isinstance(value, int):
- return False
- if value < 0:
- return False
- # XXX further validation?
- return True
- def fontInfoOpenTypeOS2FamilyClassValidator(values: Any) -> bool:
- """
- Version 2+.
- """
- if not isinstance(values, (list, tuple)):
- return False
- if len(values) != 2:
- return False
- for value in values:
- if not isinstance(value, int):
- return False
- classID, subclassID = values
- if classID < 0 or classID > 14:
- return False
- if subclassID < 0 or subclassID > 15:
- return False
- return True
- def fontInfoPostscriptBluesValidator(values: Any) -> bool:
- """
- Version 2+.
- """
- if not isinstance(values, (list, tuple)):
- return False
- if len(values) > 14:
- return False
- if len(values) % 2:
- return False
- for value in values:
- if not isinstance(value, numberTypes):
- return False
- return True
- def fontInfoPostscriptOtherBluesValidator(values: Any) -> bool:
- """
- Version 2+.
- """
- if not isinstance(values, (list, tuple)):
- return False
- if len(values) > 10:
- return False
- if len(values) % 2:
- return False
- for value in values:
- if not isinstance(value, numberTypes):
- return False
- return True
- def fontInfoPostscriptStemsValidator(values: Any) -> bool:
- """
- Version 2+.
- """
- if not isinstance(values, (list, tuple)):
- return False
- if len(values) > 12:
- return False
- for value in values:
- if not isinstance(value, numberTypes):
- return False
- return True
- def fontInfoPostscriptWindowsCharacterSetValidator(value: Any) -> bool:
- """
- Version 2+.
- """
- validValues = list(range(1, 21))
- if value not in validValues:
- return False
- return True
- def fontInfoWOFFMetadataUniqueIDValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- dictPrototype: GenericDict = dict(id=(str, True))
- if not genericDictValidator(value, dictPrototype):
- return False
- return True
- def fontInfoWOFFMetadataVendorValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- dictPrototype: GenericDict = {
- "name": (str, True),
- "url": (str, False),
- "dir": (str, False),
- "class": (str, False),
- }
- if not genericDictValidator(value, dictPrototype):
- return False
- if "dir" in value and value.get("dir") not in ("ltr", "rtl"):
- return False
- return True
- def fontInfoWOFFMetadataCreditsValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- dictPrototype: GenericDict = dict(credits=(list, True))
- if not genericDictValidator(value, dictPrototype):
- return False
- if not len(value["credits"]):
- return False
- dictPrototype = {
- "name": (str, True),
- "url": (str, False),
- "role": (str, False),
- "dir": (str, False),
- "class": (str, False),
- }
- for credit in value["credits"]:
- if not genericDictValidator(credit, dictPrototype):
- return False
- if "dir" in credit and credit.get("dir") not in ("ltr", "rtl"):
- return False
- return True
- def fontInfoWOFFMetadataDescriptionValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- dictPrototype: GenericDict = dict(url=(str, False), text=(list, True))
- if not genericDictValidator(value, dictPrototype):
- return False
- for text in value["text"]:
- if not fontInfoWOFFMetadataTextValue(text):
- return False
- return True
- def fontInfoWOFFMetadataLicenseValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- dictPrototype: GenericDict = dict(
- url=(str, False), text=(list, False), id=(str, False)
- )
- if not genericDictValidator(value, dictPrototype):
- return False
- if "text" in value:
- for text in value["text"]:
- if not fontInfoWOFFMetadataTextValue(text):
- return False
- return True
- def fontInfoWOFFMetadataTrademarkValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- dictPrototype: GenericDict = dict(text=(list, True))
- if not genericDictValidator(value, dictPrototype):
- return False
- for text in value["text"]:
- if not fontInfoWOFFMetadataTextValue(text):
- return False
- return True
- def fontInfoWOFFMetadataCopyrightValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- dictPrototype: GenericDict = dict(text=(list, True))
- if not genericDictValidator(value, dictPrototype):
- return False
- for text in value["text"]:
- if not fontInfoWOFFMetadataTextValue(text):
- return False
- return True
- def fontInfoWOFFMetadataLicenseeValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- dictPrototype: GenericDict = {
- "name": (str, True),
- "dir": (str, False),
- "class": (str, False),
- }
- if not genericDictValidator(value, dictPrototype):
- return False
- if "dir" in value and value.get("dir") not in ("ltr", "rtl"):
- return False
- return True
- def fontInfoWOFFMetadataTextValue(value: Any) -> bool:
- """
- Version 3+.
- """
- dictPrototype: GenericDict = {
- "text": (str, True),
- "language": (str, False),
- "dir": (str, False),
- "class": (str, False),
- }
- if not genericDictValidator(value, dictPrototype):
- return False
- if "dir" in value and value.get("dir") not in ("ltr", "rtl"):
- return False
- return True
- def fontInfoWOFFMetadataExtensionsValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- if not isinstance(value, list):
- return False
- if not value:
- return False
- for extension in value:
- if not fontInfoWOFFMetadataExtensionValidator(extension):
- return False
- return True
- def fontInfoWOFFMetadataExtensionValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- dictPrototype: GenericDict = dict(
- names=(list, False), items=(list, True), id=(str, False)
- )
- if not genericDictValidator(value, dictPrototype):
- return False
- if "names" in value:
- for name in value["names"]:
- if not fontInfoWOFFMetadataExtensionNameValidator(name):
- return False
- for item in value["items"]:
- if not fontInfoWOFFMetadataExtensionItemValidator(item):
- return False
- return True
- def fontInfoWOFFMetadataExtensionItemValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- dictPrototype: GenericDict = dict(
- id=(str, False), names=(list, True), values=(list, True)
- )
- if not genericDictValidator(value, dictPrototype):
- return False
- for name in value["names"]:
- if not fontInfoWOFFMetadataExtensionNameValidator(name):
- return False
- for val in value["values"]:
- if not fontInfoWOFFMetadataExtensionValueValidator(val):
- return False
- return True
- def fontInfoWOFFMetadataExtensionNameValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- dictPrototype: GenericDict = {
- "text": (str, True),
- "language": (str, False),
- "dir": (str, False),
- "class": (str, False),
- }
- if not genericDictValidator(value, dictPrototype):
- return False
- if "dir" in value and value.get("dir") not in ("ltr", "rtl"):
- return False
- return True
- def fontInfoWOFFMetadataExtensionValueValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- dictPrototype: GenericDict = {
- "text": (str, True),
- "language": (str, False),
- "dir": (str, False),
- "class": (str, False),
- }
- if not genericDictValidator(value, dictPrototype):
- return False
- if "dir" in value and value.get("dir") not in ("ltr", "rtl"):
- return False
- return True
- # ----------
- # Guidelines
- # ----------
- def guidelinesValidator(value: Any, identifiers: Optional[set[str]] = None) -> bool:
- """
- Version 3+.
- """
- if not isinstance(value, list):
- return False
- if identifiers is None:
- identifiers = set()
- for guide in value:
- if not guidelineValidator(guide):
- return False
- identifier = guide.get("identifier")
- if identifier is not None:
- if identifier in identifiers:
- return False
- identifiers.add(identifier)
- return True
- _guidelineDictPrototype: GenericDict = dict(
- x=((int, float), False),
- y=((int, float), False),
- angle=((int, float), False),
- name=(str, False),
- color=(str, False),
- identifier=(str, False),
- )
- def guidelineValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- if not genericDictValidator(value, _guidelineDictPrototype):
- return False
- angle = value.get("angle")
- # angle must be between 0 and 360
- if angle is not None:
- if angle < 0:
- return False
- if angle > 360:
- return False
- # identifier must be 1 or more characters
- identifier = value.get("identifier")
- if identifier is not None and not identifierValidator(identifier):
- return False
- # color must follow the proper format
- color = value.get("color")
- if color is not None and not colorValidator(color):
- return False
- return True
- # -------
- # Anchors
- # -------
- def anchorsValidator(value: Any, identifiers: Optional[set[str]] = None) -> bool:
- """
- Version 3+.
- """
- if not isinstance(value, list):
- return False
- if identifiers is None:
- identifiers = set()
- for anchor in value:
- if not anchorValidator(anchor):
- return False
- identifier = anchor.get("identifier")
- if identifier is not None:
- if identifier in identifiers:
- return False
- identifiers.add(identifier)
- return True
- _anchorDictPrototype: GenericDict = dict(
- x=((int, float), False),
- y=((int, float), False),
- name=(str, False),
- color=(str, False),
- identifier=(str, False),
- )
- def anchorValidator(value: Any) -> bool:
- """
- Version 3+.
- """
- if not genericDictValidator(value, _anchorDictPrototype):
- return False
- x = value.get("x")
- y = value.get("y")
- # x and y must be present
- if x is None or y is None:
- return False
- # identifier must be 1 or more characters
- identifier = value.get("identifier")
- if identifier is not None and not identifierValidator(identifier):
- return False
- # color must follow the proper format
- color = value.get("color")
- if color is not None and not colorValidator(color):
- return False
- return True
- # ----------
- # Identifier
- # ----------
- def identifierValidator(value: Any) -> bool:
- """
- Version 3+.
- >>> identifierValidator("a")
- True
- >>> identifierValidator("")
- False
- >>> identifierValidator("a" * 101)
- False
- """
- validCharactersMin = 0x20
- validCharactersMax = 0x7E
- if not isinstance(value, str):
- return False
- if not value:
- return False
- if len(value) > 100:
- return False
- for c in value:
- i = ord(c)
- if i < validCharactersMin or i > validCharactersMax:
- return False
- return True
- # -----
- # Color
- # -----
- def colorValidator(value: Any) -> bool:
- """
- Version 3+.
- >>> colorValidator("0,0,0,0")
- True
- >>> colorValidator(".5,.5,.5,.5")
- True
- >>> colorValidator("0.5,0.5,0.5,0.5")
- True
- >>> colorValidator("1,1,1,1")
- True
- >>> colorValidator("2,0,0,0")
- False
- >>> colorValidator("0,2,0,0")
- False
- >>> colorValidator("0,0,2,0")
- False
- >>> colorValidator("0,0,0,2")
- False
- >>> colorValidator("1r,1,1,1")
- False
- >>> colorValidator("1,1g,1,1")
- False
- >>> colorValidator("1,1,1b,1")
- False
- >>> colorValidator("1,1,1,1a")
- False
- >>> colorValidator("1 1 1 1")
- False
- >>> colorValidator("1 1,1,1")
- False
- >>> colorValidator("1,1 1,1")
- False
- >>> colorValidator("1,1,1 1")
- False
- >>> colorValidator("1, 1, 1, 1")
- True
- """
- if not isinstance(value, str):
- return False
- parts = value.split(",")
- if len(parts) != 4:
- return False
- for part in parts:
- part = part.strip()
- converted = False
- number: IntFloat
- try:
- number = int(part)
- converted = True
- except ValueError:
- pass
- if not converted:
- try:
- number = float(part)
- converted = True
- except ValueError:
- pass
- if not converted:
- return False
- if not 0 <= number <= 1:
- return False
- return True
- # -----
- # image
- # -----
- pngSignature: bytes = b"\x89PNG\r\n\x1a\n"
- _imageDictPrototype: GenericDict = dict(
- fileName=(str, True),
- xScale=((int, float), False),
- xyScale=((int, float), False),
- yxScale=((int, float), False),
- yScale=((int, float), False),
- xOffset=((int, float), False),
- yOffset=((int, float), False),
- color=(str, False),
- )
- def imageValidator(value):
- """
- Version 3+.
- """
- if not genericDictValidator(value, _imageDictPrototype):
- return False
- # fileName must be one or more characters
- if not value["fileName"]:
- return False
- # color must follow the proper format
- color = value.get("color")
- if color is not None and not colorValidator(color):
- return False
- return True
- def pngValidator(
- path: Optional[str] = None,
- data: Optional[bytes] = None,
- fileObj: Optional[Any] = None,
- ) -> tuple[bool, Any]:
- """
- Version 3+.
- This checks the signature of the image data.
- """
- assert path is not None or data is not None or fileObj is not None
- if path is not None:
- with open(path, "rb") as f:
- signature = f.read(8)
- elif data is not None:
- signature = data[:8]
- elif fileObj is not None:
- pos = fileObj.tell()
- signature = fileObj.read(8)
- fileObj.seek(pos)
- if signature != pngSignature:
- return False, "Image does not begin with the PNG signature."
- return True, None
- # -------------------
- # layercontents.plist
- # -------------------
- def layerContentsValidator(
- value: Any, ufoPathOrFileSystem: Union[str, fs.base.FS]
- ) -> tuple[bool, Optional[str]]:
- """
- Check the validity of layercontents.plist.
- Version 3+.
- """
- if isinstance(ufoPathOrFileSystem, fs.base.FS):
- fileSystem = ufoPathOrFileSystem
- else:
- fileSystem = fs.osfs.OSFS(ufoPathOrFileSystem)
- bogusFileMessage = "layercontents.plist in not in the correct format."
- # file isn't in the right format
- if not isinstance(value, list):
- return False, bogusFileMessage
- # work through each entry
- usedLayerNames = set()
- usedDirectories = set()
- contents = {}
- for entry in value:
- # layer entry in the incorrect format
- if not isinstance(entry, list):
- return False, bogusFileMessage
- if not len(entry) == 2:
- return False, bogusFileMessage
- for i in entry:
- if not isinstance(i, str):
- return False, bogusFileMessage
- layerName, directoryName = entry
- # check directory naming
- if directoryName != "glyphs":
- if not directoryName.startswith("glyphs."):
- return (
- False,
- "Invalid directory name (%s) in layercontents.plist."
- % directoryName,
- )
- if len(layerName) == 0:
- return False, "Empty layer name in layercontents.plist."
- # directory doesn't exist
- if not fileSystem.exists(directoryName):
- return False, "A glyphset does not exist at %s." % directoryName
- # default layer name
- if layerName == "public.default" and directoryName != "glyphs":
- return (
- False,
- "The name public.default is being used by a layer that is not the default.",
- )
- # check usage
- if layerName in usedLayerNames:
- return (
- False,
- "The layer name %s is used by more than one layer." % layerName,
- )
- usedLayerNames.add(layerName)
- if directoryName in usedDirectories:
- return (
- False,
- "The directory %s is used by more than one layer." % directoryName,
- )
- usedDirectories.add(directoryName)
- # store
- contents[layerName] = directoryName
- # missing default layer
- foundDefault = "glyphs" in contents.values()
- if not foundDefault:
- return False, "The required default glyph set is not in the UFO."
- return True, None
- # ------------
- # groups.plist
- # ------------
- def groupsValidator(value: Any) -> tuple[bool, Optional[str]]:
- """
- Check the validity of the groups.
- Version 3+ (though it's backwards compatible with UFO 1 and UFO 2).
- >>> groups = {"A" : ["A", "A"], "A2" : ["A"]}
- >>> groupsValidator(groups)
- (True, None)
- >>> groups = {"" : ["A"]}
- >>> valid, msg = groupsValidator(groups)
- >>> valid
- False
- >>> print(msg)
- A group has an empty name.
- >>> groups = {"public.awesome" : ["A"]}
- >>> groupsValidator(groups)
- (True, None)
- >>> groups = {"public.kern1." : ["A"]}
- >>> valid, msg = groupsValidator(groups)
- >>> valid
- False
- >>> print(msg)
- The group data contains a kerning group with an incomplete name.
- >>> groups = {"public.kern2." : ["A"]}
- >>> valid, msg = groupsValidator(groups)
- >>> valid
- False
- >>> print(msg)
- The group data contains a kerning group with an incomplete name.
- >>> groups = {"public.kern1.A" : ["A"], "public.kern2.A" : ["A"]}
- >>> groupsValidator(groups)
- (True, None)
- >>> groups = {"public.kern1.A1" : ["A"], "public.kern1.A2" : ["A"]}
- >>> valid, msg = groupsValidator(groups)
- >>> valid
- False
- >>> print(msg)
- The glyph "A" occurs in too many kerning groups.
- """
- bogusFormatMessage = "The group data is not in the correct format."
- if not isDictEnough(value):
- return False, bogusFormatMessage
- firstSideMapping: dict[str, str] = {}
- secondSideMapping: dict[str, str] = {}
- for groupName, glyphList in value.items():
- if not isinstance(groupName, (str)):
- return False, bogusFormatMessage
- if not isinstance(glyphList, (list, tuple)):
- return False, bogusFormatMessage
- if not groupName:
- return False, "A group has an empty name."
- if groupName.startswith("public."):
- if not groupName.startswith("public.kern1.") and not groupName.startswith(
- "public.kern2."
- ):
- # unknown public.* name. silently skip.
- continue
- else:
- if len("public.kernN.") == len(groupName):
- return (
- False,
- "The group data contains a kerning group with an incomplete name.",
- )
- if groupName.startswith("public.kern1."):
- d = firstSideMapping
- else:
- d = secondSideMapping
- for glyphName in glyphList:
- if not isinstance(glyphName, str):
- return (
- False,
- "The group data %s contains an invalid member." % groupName,
- )
- if glyphName in d:
- return (
- False,
- 'The glyph "%s" occurs in too many kerning groups.' % glyphName,
- )
- d[glyphName] = groupName
- return True, None
- # -------------
- # kerning.plist
- # -------------
- def kerningValidator(data: Any) -> tuple[bool, Optional[str]]:
- """
- Check the validity of the kerning data structure.
- Version 3+ (though it's backwards compatible with UFO 1 and UFO 2).
- >>> kerning = {"A" : {"B" : 100}}
- >>> kerningValidator(kerning)
- (True, None)
- >>> kerning = {"A" : ["B"]}
- >>> valid, msg = kerningValidator(kerning)
- >>> valid
- False
- >>> print(msg)
- The kerning data is not in the correct format.
- >>> kerning = {"A" : {"B" : "100"}}
- >>> valid, msg = kerningValidator(kerning)
- >>> valid
- False
- >>> print(msg)
- The kerning data is not in the correct format.
- """
- bogusFormatMessage = "The kerning data is not in the correct format."
- if not isinstance(data, Mapping):
- return False, bogusFormatMessage
- for first, secondDict in data.items():
- if not isinstance(first, str):
- return False, bogusFormatMessage
- elif not isinstance(secondDict, Mapping):
- return False, bogusFormatMessage
- for second, value in secondDict.items():
- if not isinstance(second, str):
- return False, bogusFormatMessage
- elif not isinstance(value, numberTypes):
- return False, bogusFormatMessage
- return True, None
- # -------------
- # lib.plist/lib
- # -------------
- _bogusLibFormatMessage = "The lib data is not in the correct format: %s"
- def fontLibValidator(value: Any) -> tuple[bool, Optional[str]]:
- """
- Check the validity of the lib.
- Version 3+ (though it's backwards compatible with UFO 1 and UFO 2).
- >>> lib = {"foo" : "bar"}
- >>> fontLibValidator(lib)
- (True, None)
- >>> lib = {"public.awesome" : "hello"}
- >>> fontLibValidator(lib)
- (True, None)
- >>> lib = {"public.glyphOrder" : ["A", "C", "B"]}
- >>> fontLibValidator(lib)
- (True, None)
- >>> lib = "hello"
- >>> valid, msg = fontLibValidator(lib)
- >>> valid
- False
- >>> print(msg) # doctest: +ELLIPSIS
- The lib data is not in the correct format: expected a dictionary, ...
- >>> lib = {1: "hello"}
- >>> valid, msg = fontLibValidator(lib)
- >>> valid
- False
- >>> print(msg)
- The lib key is not properly formatted: expected str, found int: 1
- >>> lib = {"public.glyphOrder" : "hello"}
- >>> valid, msg = fontLibValidator(lib)
- >>> valid
- False
- >>> print(msg) # doctest: +ELLIPSIS
- public.glyphOrder is not properly formatted: expected list or tuple,...
- >>> lib = {"public.glyphOrder" : ["A", 1, "B"]}
- >>> valid, msg = fontLibValidator(lib)
- >>> valid
- False
- >>> print(msg) # doctest: +ELLIPSIS
- public.glyphOrder is not properly formatted: expected str,...
- """
- if not isDictEnough(value):
- reason = "expected a dictionary, found %s" % type(value).__name__
- return False, _bogusLibFormatMessage % reason
- for key, value in value.items():
- if not isinstance(key, str):
- return False, (
- "The lib key is not properly formatted: expected str, found %s: %r"
- % (type(key).__name__, key)
- )
- # public.glyphOrder
- if key == "public.glyphOrder":
- bogusGlyphOrderMessage = "public.glyphOrder is not properly formatted: %s"
- if not isinstance(value, (list, tuple)):
- reason = "expected list or tuple, found %s" % type(value).__name__
- return False, bogusGlyphOrderMessage % reason
- for glyphName in value:
- if not isinstance(glyphName, str):
- reason = "expected str, found %s" % type(glyphName).__name__
- return False, bogusGlyphOrderMessage % reason
- return True, None
- # --------
- # GLIF lib
- # --------
- def glyphLibValidator(value: Any) -> tuple[bool, Optional[str]]:
- """
- Check the validity of the lib.
- Version 3+ (though it's backwards compatible with UFO 1 and UFO 2).
- >>> lib = {"foo" : "bar"}
- >>> glyphLibValidator(lib)
- (True, None)
- >>> lib = {"public.awesome" : "hello"}
- >>> glyphLibValidator(lib)
- (True, None)
- >>> lib = {"public.markColor" : "1,0,0,0.5"}
- >>> glyphLibValidator(lib)
- (True, None)
- >>> lib = {"public.markColor" : 1}
- >>> valid, msg = glyphLibValidator(lib)
- >>> valid
- False
- >>> print(msg)
- public.markColor is not properly formatted.
- """
- if not isDictEnough(value):
- reason = "expected a dictionary, found %s" % type(value).__name__
- return False, _bogusLibFormatMessage % reason
- for key, value in value.items():
- if not isinstance(key, str):
- reason = "key (%s) should be a string" % key
- return False, _bogusLibFormatMessage % reason
- # public.markColor
- if key == "public.markColor":
- if not colorValidator(value):
- return False, "public.markColor is not properly formatted."
- return True, None
- if __name__ == "__main__":
- import doctest
- doctest.testmod()
|