From f86be23d1567fb0b36d96c84a6118ee884ee40ee Mon Sep 17 00:00:00 2001
From: Thomas Kluyver <thomas@kluyver.me.uk>
Date: Fri, 16 Dec 2022 17:47:43 +0100
Subject: [PATCH] Add support for saving metadata fragments & merging into
 calibration_metadata.yml

---
 src/cal_tools/tools.py | 31 +++++++++++++++++++++++++++++++
 1 file changed, 31 insertions(+)

diff --git a/src/cal_tools/tools.py b/src/cal_tools/tools.py
index 2d15866ac..0760d48a7 100644
--- a/src/cal_tools/tools.py
+++ b/src/cal_tools/tools.py
@@ -10,6 +10,7 @@ from os import environ, listdir, path
 from os.path import isfile
 from pathlib import Path
 from queue import Queue
+from tempfile import NamedTemporaryFile
 from time import sleep
 from typing import List, Optional, Tuple, Union
 from urllib.parse import urljoin
@@ -811,6 +812,14 @@ def module_index_to_qm(index: int, total_modules: int = 16):
     return f"Q{quad+1}M{mod+1}"
 
 
+def recursive_update(target: dict, source: dict):
+    for k, v2 in source.items():
+        v1 = target.get(k, None)
+        if isinstance(v1, dict) and isinstance(v2, dict):
+            recursive_update(v1, v2)
+        else:
+            target[k] = v2
+
 class CalibrationMetadata(dict):
     """Convenience class: dictionary stored in metadata YAML file
 
@@ -847,6 +856,28 @@ class CalibrationMetadata(dict):
         with (copy_dir / self._yaml_fn.name).open("w") as fd:
             yaml.safe_dump(dict(self), fd)
 
+    def add_fragment(self, data: dict):
+        """Save metadata to a separate 'fragment' file to be merged later
+
+        Avoids a risk of corrupting the main file by writing in parallel.
+        """
+        with NamedTemporaryFile("w", dir=self._yaml_fn.parent,
+                    prefix='metadata_frag_', suffix='.yml', delete=False) as fd:
+            yaml.safe_dump(data, fd)
+
+    def gather_fragments(self):
+        """Merge in fragments saved by add_fragment(), then delete them"""
+        frag_files = list(self._yaml_fn.parent.glob('metadata_frag_*.yml'))
+        for fn in frag_files:
+            with fn.open("r") as fd:
+                data = yaml.safe_load(fd)
+                recursive_update(self, data)
+
+        self.save()
+
+        for fn in frag_files:
+            fn.unlink()
+
 
 def save_constant_metadata(
     retrieved_constants: dict,
-- 
GitLab