From 5b670f85446c8d80b883a625d1e1b8b86bf86193 Mon Sep 17 00:00:00 2001 From: Just van Rossum Date: Wed, 13 Dec 2023 21:37:40 +0100 Subject: [PATCH 1/2] Don't return an error string, just raise an exception --- src/fontra_rcjk/backend_mysql.py | 92 +++++++++++++++----------------- 1 file changed, 44 insertions(+), 48 deletions(-) diff --git a/src/fontra_rcjk/backend_mysql.py b/src/fontra_rcjk/backend_mysql.py index 489c866..5f42235 100644 --- a/src/fontra_rcjk/backend_mysql.py +++ b/src/fontra_rcjk/backend_mysql.py @@ -27,7 +27,6 @@ standardCustomDataItems, unpackAxes, ) -from .client import HTTPError logger = logging.getLogger(__name__) @@ -46,6 +45,10 @@ } +class RCJKEditError(Exception): + pass + + @dataclass class RCJKGlyphInfo: typeCode: str @@ -268,7 +271,9 @@ async def putGlyph( self._writingChanges -= 1 logger.info(f"Done writing {glyphName}") - async def _putGlyph(self, glyphName, glyph, unicodes): + async def _putGlyph( + self, glyphName: str, glyph: VariableGlyph, unicodes: list[int] + ) -> None: defaultLocation = await self.getDefaultLocation() if glyphName not in self._rcjkGlyphInfo: @@ -285,62 +290,53 @@ async def _putGlyph(self, glyphName, glyph, unicodes): self._glyphMap[glyphName] = unicodes - try: - lockResponse = await self._callGlyphMethod( - glyphName, "lock", return_data=False - ) - except HTTPError as error: - return f"Can't lock glyph ({error})" - - errorMessage = None + lockResponse = await self._callGlyphMethod(glyphName, "lock", return_data=False) try: glyphTimeStamp = self._glyphTimeStamps[glyphName] currentTimeStamp = getUpdatedTimeStamp(lockResponse["data"]) if glyphTimeStamp != currentTimeStamp: - errorMessage = "Someone else made an edit just before you." - else: - for layerName, layerGlyph in layerGlyphs.items(): - xmlData = layerGlyph.asGLIFData() - existingXMLData = existingLayerData.get(layerName) - if xmlData == existingXMLData: - # There was no change in the xml data, skip the update - continue - if layerName == "foreground": - args = (glyphName, "update", xmlData) - else: - methodName = "layer_update" - if existingXMLData is None: - logger.info(f"Creating layer {layerName} of {glyphName}") - methodName = "layer_create" - args = (glyphName, methodName, layerName, xmlData) - await self._callGlyphMethod( - *args, - return_data=False, - return_layers=False, - ) - for layerName in set(existingLayerData) - set(layerGlyphs): - logger.info(f"Deleting layer {layerName} of {glyphName}") - await self._callGlyphMethod( - glyphName, - "layer_delete", - layerName, - ) - self._glyphCache[glyphName] = layerGlyphs + raise RCJKEditError("Someone else made an edit just before you.") + + for layerName, layerGlyph in layerGlyphs.items(): + xmlData = layerGlyph.asGLIFData() + existingXMLData = existingLayerData.get(layerName) + if xmlData == existingXMLData: + # There was no change in the xml data, skip the update + continue + if layerName == "foreground": + args = [glyphName, "update", xmlData] + else: + methodName = "layer_update" + if existingXMLData is None: + logger.info(f"Creating layer {layerName} of {glyphName}") + methodName = "layer_create" + args = [glyphName, methodName, layerName, xmlData] + await self._callGlyphMethod( + *args, + return_data=False, + return_layers=False, + ) + for layerName in set(existingLayerData) - set(layerGlyphs): + logger.info(f"Deleting layer {layerName} of {glyphName}") + await self._callGlyphMethod( + glyphName, + "layer_delete", + layerName, + ) + self._glyphCache[glyphName] = layerGlyphs + except Exception: + asyncio.get_running_loop().call_soon(self._pollNowEvent.set) + raise finally: unlockResponse = await self._callGlyphMethod( glyphName, "unlock", return_data=False ) - if errorMessage: - asyncio.get_running_loop().call_soon(self._pollNowEvent.set) - if errorMessage is None: - timeStamp = getUpdatedTimeStamp(unlockResponse["data"]) - self._glyphTimeStamps[glyphName] = timeStamp - self._rcjkGlyphInfo[glyphName].updated = timeStamp - self._writeGlyphToCacheDir(glyphName, glyph) - - return errorMessage + timeStamp = getUpdatedTimeStamp(unlockResponse["data"]) + self._glyphTimeStamps[glyphName] = timeStamp + self._rcjkGlyphInfo[glyphName].updated = timeStamp + self._writeGlyphToCacheDir(glyphName, glyph) async def _newGlyph(self, glyphName: str, unicodes: list[int]) -> None: # In _newGlyph() we create a new character glyph in the database. From 6e11b336d302611a5d82987ce28f253f7ccd4d00 Mon Sep 17 00:00:00 2001 From: Just van Rossum Date: Wed, 13 Dec 2023 21:53:09 +0100 Subject: [PATCH 2/2] on error, set poll now only after unlock succeeded --- src/fontra_rcjk/backend_mysql.py | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/src/fontra_rcjk/backend_mysql.py b/src/fontra_rcjk/backend_mysql.py index 5f42235..31f3adb 100644 --- a/src/fontra_rcjk/backend_mysql.py +++ b/src/fontra_rcjk/backend_mysql.py @@ -292,6 +292,8 @@ async def _putGlyph( lockResponse = await self._callGlyphMethod(glyphName, "lock", return_data=False) + error = False + try: glyphTimeStamp = self._glyphTimeStamps[glyphName] currentTimeStamp = getUpdatedTimeStamp(lockResponse["data"]) @@ -326,12 +328,14 @@ async def _putGlyph( ) self._glyphCache[glyphName] = layerGlyphs except Exception: - asyncio.get_running_loop().call_soon(self._pollNowEvent.set) + error = True raise finally: unlockResponse = await self._callGlyphMethod( glyphName, "unlock", return_data=False ) + if error: + asyncio.get_running_loop().call_soon(self._pollNowEvent.set) timeStamp = getUpdatedTimeStamp(unlockResponse["data"]) self._glyphTimeStamps[glyphName] = timeStamp