Source code for pymepix.main

# This file is part of Pymepix
#
# In all scientific work using Pymepix, please reference it as
#
# A. F. Al-Refaie, M. Johny, J. Correa, D. Pennicard, P. Svihra, A. Nomerotski, S. Trippel, and J. Küpper:
# "PymePix: a python library for SPIDR readout of Timepix3", J. Inst. 14, P10003 (2019)
# https://doi.org/10.1088/1748-0221/14/10/P10003
# https://arxiv.org/abs/1905.07999
#
# Pymepix is free software: you can redistribute it and/or modify it under the terms of the GNU
# General Public License as published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
# See the GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License along with this program.
# If not, see <https://www.gnu.org/licenses/>.

"""Main module for pymepix"""

import argparse
import json
import logging
import os
import re
import time

#import multiprocessing
#multiprocessing.set_start_method("fork", force=True)

# from .api.api import make_app
from tornado.ioloop import IOLoop
from tornado.web import Application, HTTPError, RequestHandler

import pymepix.config.load_config as cfg
from pymepix.core.log import Logger
from pymepix.post_processing import run_post_processing
from pymepix.processing.acquisition import CentroidPipeline
from pymepix.pymepix_connection import PymepixConnection

# from .api.api import make_app


log = Logger("main")
# logging.basicConfig(
#     level=logging.INFO,
#     format="%(asctime)s - %(name)s - %(levelname)s - %(message)s",
# )
# log.setLevel(logging.DEBUG)


[docs] def connect_timepix(args): if not os.path.exists(args.output): # Connect to Timepix pymepix = PymepixConnection() # If there are no valid timepix detected then quit() if len(pymepix) == 0: log.error("-------ERROR: SPIDR FOUND BUT NO VALID TIMEPIX DEVICE DETECTED ---------- ") quit() if args.spx: log.info(f"Opening Sophy file {args.spx}") pymepix[0].loadConfig(args.spx) # Switch to TOF mode if set if args.decode and args.tof: pymepix[0].acquisition.enableEvents = True # Set the bias voltage pymepix.biasVoltage = args.bias # self._timepix._spidr.resetTimers() # self._timepix._spidr.restartTimers() # time.sleep(1) # give camera time to reset timers # Start acquisition pymepix.start() pymepix._timepix_devices[0].start_recording(args.output) time.sleep(args.time) pymepix._timepix_devices[0].stop_recording() pymepix.stop() else: log.info(f"Outputfile {args.output} already exists. Please make sure the specified file does not exist.")
[docs] def post_process(args): run_post_processing( args.file.name, args.output_file, args.number_of_processes, args.timewalk_file, args.cent_timewalk_file, args.cam_gen, )
[docs] def is_jsonable(x): try: json.dumps(x) return True except (TypeError, OverflowError): return False
[docs] def get_path(ref, path): try: for tkn in path: if tkn[0] != "[": ref = getattr(ref, tkn) else: ref = ref[int(tkn[1:-1])] except Exception as e: raise HTTPError(400, f"Bad request {str(e)}") return ref
[docs] class RootHandler(RequestHandler):
[docs] def get(self): self.write({"message": "Online"})
[docs] class TPXpropertyHandler(RequestHandler):
[docs] def get(self): global timepix_obj arguments = self.get_arguments("param_name") tkns = re.findall(r"[\w']+|\[\d+\]", arguments[0]) ref = timepix_obj ref = get_path(ref, tkns) if is_jsonable(ref): self.write({arguments[0]: ref}) else: raise HTTPError(405, "Parameter value is not JSON serializable")
[docs] def post(self): global timepix_obj try: data = json.loads(self.request.body) except Exception as e: raise HTTPError(400, f"Bad request {str(e)}") for key, val in data.items(): ref = timepix_obj tkns = re.findall(r"[\w']+|\[\d+\]", key) ref = get_path(ref, tkns[:-1]) if tkns[-1][0] != "[": setattr(ref, tkns[-1], val) else: ref[int(tkns[-1][1:-1])] = val self.write(data)
[docs] class TPXmethodHandler(RequestHandler):
[docs] def post(self): global timepix_obj try: data = json.loads(self.request.body) except json.JSONDecodeError as e: raise HTTPError(400, f"Bad request: {str(e)}") except Exception as e: log.error(f"Unexpected error occurred: {str(e)}", exc_info=True) raise HTTPError(400, "An unexpected error occurred during post-processing") try: func_name = data["func_name"] tkns = re.findall(r"[\w']+|\[\d+\]", func_name) ref = timepix_obj ref = get_path(ref, tkns) data.pop("func_name") res = ref(**data) if is_jsonable(res): self.write({"result": res}) else: self.write({"result": "Result is not JSON serializable"}) except Exception as e: raise HTTPError(400, f"Bad request: {str(e)}")
[docs] class PostprocessHandler(RequestHandler):
[docs] def post(self): try: data = json.loads(self.request.body) except json.JSONDecodeError as e: raise HTTPError(400, f"Bad request: {str(e)}") except Exception as e: log.error(f"Unexpected error occurred: {str(e)}", exc_info=True) raise HTTPError(400, "An unexpected error occurred during post-processing") try: run_post_processing(**data) except (ValueError, TypeError, KeyError) as e: raise HTTPError(400, f"Bad request: {str(e)}") except Exception as e: log.error(f"Unexpected error occurred: {str(e)}", exc_info=True) raise HTTPError(400, "An unexpected error occurred during post-processing")
[docs] def make_app(): urls = [ ("/", RootHandler), (r"/tpxproperty", TPXpropertyHandler), (r"/tpxmethod", TPXmethodHandler), (r"/postprocess", PostprocessHandler), ] return Application(urls, debug=True)
[docs] def start_api(args): global timepix_obj logging.getLogger("tornado").setLevel(logging.ERROR) timepix_obj = PymepixConnection( spidr_address=(args.ip, args.port), camera_generation=args.cam_gen, pipeline_class=CentroidPipeline ) if len(timepix_obj) == 0: logging.error("-------ERROR: SPIDR FOUND BUT NO VALID TIMEPIX DEVICE DETECTED ---------- ") quit() if args.spx: logging.info(f"Opening Sophy file {args.spx}") timepix_obj[0].loadConfig(args.spx) # Switch to TOF mode if set if args.decode and args.tof: timepix_obj[0].acquisition.enableEvents = True # Set the bias voltage timepix_obj.biasVoltage = args.bias app = make_app() app.listen(args.api_port) IOLoop.instance().start()
[docs] def main(): parser = argparse.ArgumentParser(description="Timepix acquisition script") subparsers = parser.add_subparsers( description="Processing type", help="Select which type of process should be executed", required=True, dest="command", ) parser_connect = subparsers.add_parser("connect", help="Connect to TimePix camera and acquire data.") parser_connect.set_defaults(func=connect_timepix) parser_connect.add_argument( "-i", "--ip", dest="ip", type=str, default=cfg.default_cfg["timepix"]["ip"], help="IP address of Timepix", ) parser_connect.add_argument( "-p", "--port", dest="port", type=int, default=50000, help="TCP port to use for the connection", ) parser_connect.add_argument("-s", "--spx", dest="spx", type=str, help="Sophy config file to load") parser_connect.add_argument( "-v", "--bias", dest="bias", type=float, default=50, help="Bias voltage in Volts", ) parser_connect.add_argument( "-t", "--time", dest="time", type=float, help="Acquisition time in seconds", required=True, ) parser_connect.add_argument( "-o", "--output", dest="output", type=str, help="output filename prefix", required=True, ) parser_connect.add_argument( "-d", "--decode", dest="decode", type=bool, help="Store decoded values instead", default=False, ) parser_connect.add_argument( "-T", "--tof", dest="tof", type=bool, help="Compute TOF if decode is enabled", default=False, ) parser_connect.add_argument( "-c", "--config", dest="cfg", type=str, default="default.yaml", help="Config file", ) parser_connect.add_argument( "-g", "--cam_gen", dest="cam_gen", type=int, default=3, help="Camera generation", ) parser_post_process = subparsers.add_parser( "post-process", help="Perform post-processing with a acquired raw data file." ) parser_post_process.set_defaults(func=post_process) parser_post_process.add_argument( "-f", "--file", dest="file", type=argparse.FileType("rb"), help="Raw data file for postprocessing", required=True, ) parser_post_process.add_argument( "-o", "--output_file", dest="output_file", type=str, help="Filename where the processed data is stored", required=True, ) parser_post_process.add_argument( "-t", "--timewalk_file", dest="timewalk_file", type=argparse.FileType("rb"), help="File containing the time walk information", ) parser_post_process.add_argument( "-c", "--cent_timewalk_file", dest="cent_timewalk_file", type=argparse.FileType("rb"), help="File containing the centroided time walk information", ) parser_post_process.add_argument( "-n", "--number_of_processes", dest="number_of_processes", type=int, default=4, help="The number of processes used for the centroiding (default: 1 => parallel processing disabled')", ) parser_post_process.add_argument( "--config", dest="cfg", type=str, default="default.yaml", help="Config file", ) parser_post_process.add_argument( "-g", "--cam_gen", dest="cam_gen", type=int, default=3, help="Camera generation", ) parser_api_service = subparsers.add_parser("api-service", help="start api service.") parser_api_service.set_defaults(func=start_api) parser_api_service.add_argument( "-i", "--ip", dest="ip", type=str, default=cfg.default_cfg["timepix"]["ip"], help="IP address of Timepix", ) parser_api_service.add_argument( "-p", "--port", dest="port", type=int, default=50000, help="TCP port to use for the connection", ) parser_api_service.add_argument( "-api_port", "--api_port", dest="api_port", type=int, default=8080, help="TCP port to use for API", ) parser_api_service.add_argument( "--config", dest="cfg", type=str, default="default.yaml", help="Config file", ) parser_api_service.add_argument("-s", "--spx", dest="spx", type=str, help="Sophy config file to load") parser_api_service.add_argument( "-d", "--decode", dest="decode", type=bool, help="Store decoded values instead", default=False, ) parser_api_service.add_argument( "-T", "--tof", dest="tof", type=bool, help="Compute TOF if decode is enabled", default=False, ) parser_api_service.add_argument( "-v", "--bias", dest="bias", type=float, default=50, help="Bias voltage in Volts", ) parser_api_service.add_argument( "-g", "--cam_gen", dest="cam_gen", type=int, default=3, help="Camera generation", ) parser_api_service.add_argument( "-pl", "--pipeline", dest="pixel_pipeline", type=str, default="pixel", help="Processing pipeline, options: centroid, pixel. Default - pixel", ) args = parser.parse_args() print(args) cfg.load_config(args.cfg) args.func(args)
if __name__ == "__main__": main()