From e7d60843f771e87e6daa11fd7831f31771f2a2c6 Mon Sep 17 00:00:00 2001 From: Cosimo Lupo Date: Wed, 31 Jul 2024 16:43:51 +0100 Subject: [PATCH 1/2] classes_test: repro LayersIterator error with 'orphan' GSGlyph reproduces https://github.com/googlefonts/glyphsLib/issues/1013 --- tests/classes_test.py | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/tests/classes_test.py b/tests/classes_test.py index 5f335cc9e..567f11c56 100755 --- a/tests/classes_test.py +++ b/tests/classes_test.py @@ -154,6 +154,17 @@ def test_append_layer_same_id(self): font.masters.append(master2) assert len({m.id for m in font.masters}) == 2 + def test_iterate_layers_of_orphan_glyph(self): + # https://github.com/googlefonts/glyphsLib/issues/1013 + glyph = GSGlyph() + assert glyph.parent is None + layer = GSLayer() + glyph.layers.append(layer) + assert layer.parent is glyph + # this ought not to raise a `KeyError: 0` exception + layers = list(glyph.layers) + assert layers[0] is layer + class GSFontTest(unittest.TestCase): def test_init(self): From 6c0d2a24872109342aac19420ea9f129cb656353 Mon Sep 17 00:00:00 2001 From: Cosimo Lupo Date: Wed, 31 Jul 2024 16:48:30 +0100 Subject: [PATCH 2/2] rewrite and simplify LayersIterator, fix issue with orphan glyph Fixes #1013 --- Lib/glyphsLib/classes.py | 62 +++++++++++++++++----------------------- 1 file changed, 26 insertions(+), 36 deletions(-) diff --git a/Lib/glyphsLib/classes.py b/Lib/glyphsLib/classes.py index 0e9e5c33c..3ccde3a91 100755 --- a/Lib/glyphsLib/classes.py +++ b/Lib/glyphsLib/classes.py @@ -520,12 +520,13 @@ def _get_by_name(self, name): class LayersIterator: - __slots__ = "curInd", "_owner", "_orderedLayers" + __slots__ = ("_layers",) def __init__(self, owner): - self.curInd = 0 - self._owner = owner - self._orderedLayers = None + if owner.parent: + self._layers = self.orderedLayers(owner) + else: + self._layers = iter(owner._layers.values()) def __iter__(self): return self @@ -534,39 +535,28 @@ def next(self): return self.__next__() def __next__(self): - if self._owner.parent: - if self.curInd >= len(self._owner.layers): - raise StopIteration - item = self.orderedLayers[self.curInd] - else: - if self.curInd >= len(self._owner._layers): - raise StopIteration - item = self._owner._layers[self.curInd] - self.curInd += 1 - return item + return next(self._layers) - @property - def orderedLayers(self): - if not self._orderedLayers: - glyphLayerIds = [ - l.associatedMasterId - for l in self._owner._layers.values() - if l.associatedMasterId == l.layerId - ] - masterIds = [m.id for m in self._owner.parent.masters] - intersectedLayerIds = set(glyphLayerIds) & set(masterIds) - orderedLayers = [ - self._owner._layers[m.id] - for m in self._owner.parent.masters - if m.id in intersectedLayerIds - ] - orderedLayers += [ - self._owner._layers[l.layerId] - for l in self._owner._layers.values() - if l.layerId not in intersectedLayerIds - ] - self._orderedLayers = orderedLayers - return self._orderedLayers + @staticmethod + def orderedLayers(glyph): + font = glyph.parent + assert font is not None + glyphLayerIds = { + l.associatedMasterId + for l in glyph._layers.values() + if l.associatedMasterId == l.layerId + } + masterIds = {m.id for m in font.masters} + intersectedLayerIds = glyphLayerIds & masterIds + orderedLayers = [ + glyph._layers[m.id] for m in font.masters if m.id in intersectedLayerIds + ] + orderedLayers.extend( + glyph._layers[l.layerId] + for l in glyph._layers.values() + if l.layerId not in intersectedLayerIds + ) + return iter(orderedLayers) class FontFontMasterProxy(Proxy):