From bdd73207b552d7d2d7a00911885e144fc798e8b8 Mon Sep 17 00:00:00 2001 From: Roy Lane Date: Thu, 9 Jan 2025 12:23:35 -0500 Subject: [PATCH] add JSON ArgumentsEncoder class to handle non-serializable values in arguments data structure --- scubagoggles/orchestrator.py | 43 ++++++++++++++++++++++++++++++++++-- 1 file changed, 41 insertions(+), 2 deletions(-) diff --git a/scubagoggles/orchestrator.py b/scubagoggles/orchestrator.py index 310f204b..ef9af07f 100644 --- a/scubagoggles/orchestrator.py +++ b/scubagoggles/orchestrator.py @@ -11,6 +11,7 @@ import shutil import webbrowser +from collections.abc import Callable from datetime import datetime, timezone from pathlib import Path from tqdm import tqdm @@ -31,6 +32,44 @@ class UserRuntimeError(RuntimeError): to do with the code). """ + +class ArgumentsEncoder(json.JSONEncoder): + + """Custom JSON encoder for the ScubaGoggles command line arguments data + structure. + """ + + def default(self, o): + + """Handles special encoding for certain fields in the arguments + data structure. + + For example, there are some fields in the arguments for internal + use, such as the dispatch function. This is not serializable with + the default encoder. + + The Path values are converted to strings. + + Any data types not handled by this method are processed by the base + class method. + + :param o: data (arbitrary type) to be serialized. + + :return: a serializable version of the given object. + """ + + # The "dispatch" value is a function and does not need to be + # serialized. + + if isinstance(o, Callable): + return None + + if isinstance(o, Path): + return str(o) + + return super().default(o) + + class Orchestrator: """The Orchestrator class runs the provider to get the GWS configuration @@ -316,7 +355,7 @@ def _run_reporter(self): param_out_path = out_folder / 'args_param.json' with param_out_path.open(mode='w', encoding='UTF-8') as parm_file: - json.dump(args_dict, parm_file, indent=4) + json.dump(args_dict, parm_file, indent=4, cls=ArgumentsEncoder) # Create the ScubaResults files scuba_results_file = out_folder / f'{args.outputproviderfilename}.json' @@ -398,7 +437,7 @@ def _run_cached(self): args_dict = vars(args) param_out_path = os.path.join(args.outputpath, 'args_param.json') with open(param_out_path, mode='w', encoding='UTF-8') as parm_file: - json.dump( args_dict, parm_file, indent=4) + json.dump(args_dict, parm_file, indent=4, cls=ArgumentsEncoder) self._rego_eval() self._run_reporter()