diff --git a/src/xfel_calibrate/calibrate.py b/src/xfel_calibrate/calibrate.py
index 655bc6cad85193be88d5f9f272d466953b86d5e9..bc43ac42465558005380f4f6b1d3febd78045791 100755
--- a/src/xfel_calibrate/calibrate.py
+++ b/src/xfel_calibrate/calibrate.py
@@ -583,7 +583,11 @@ def run(argv=None):
 
     title = title.rstrip()
 
-    run_uuid = f"t{datetime.now().strftime('%y%m%d_%H%M%S.%f')}"
+    # request_time is in local timezone
+    if args["request_time"] == "Now":
+        request_time = datetime.now()
+    else:
+        request_time = datetime.fromisoformat(args["request_time"])
 
     # check if concurrency parameter is given and we run concurrently
     if concurrency_par is not None and not any(
@@ -596,8 +600,7 @@ def run(argv=None):
     default_params_by_name = {p.name: p.value for p in nb_details.default_params}
     if 'cluster_profile' in default_params_by_name:
         if args.get("cluster_profile") == default_params_by_name["cluster_profile"]:
-            args['cluster_profile'] = "slurm_prof_{}".format(run_uuid)
-
+            args['cluster_profile'] = f"slurm_prof_{request_time:%y%m%d_%H%M%S.%f}"
 
     # wait on all jobs to run and then finalize the run by creating a report from the notebooks
     out_path = Path(default_report_path) / nb_details.detector / nb_details.caltype / datetime.now().isoformat()
@@ -609,26 +612,22 @@ def run(argv=None):
 
     out_path.mkdir(parents=True, exist_ok=True)
 
-    # Use given report name, falling back to notebook title
+    # Use given report name, or automatic unique name if not specified
+    det_name = args.get('karabo_id', nb_details.detector)
+    unique_name = f"{det_name}-{nb_details.caltype}-{request_time:%y%m%d_%H%M%S.%f}"
     if args['skip_report']:
         report_to = ''
     elif args["report_to"] is None:
-        report_to = out_path / title.replace(" ", "")
+        report_to = out_path / f"{unique_name}.pdf"
         print(f"report_to not specified, will use {report_to}")
     else:
-        report_to = Path(args["report_to"])
-        if report_to.is_dir():
-            print(f"report_to is directory, will use title '{title}' for filename")
-            report_to = report_to / title.replace(" ", "")
-        elif len(report_to.parts) == 1:
-            print(f"report_to path contained no path, saving report in '{out_path}'")
-            report_to = out_path / report_to
-
-    workdir_name = f"slurm_out_{nb_details.detector}_{nb_details.caltype}_{run_uuid}"
+        report_to = Path(args["report_to"]).with_suffix('.pdf').absolute()
+
     if report_to:
-        cal_work_dir = report_to.parent / workdir_name
+        # Work dir matching report file but without .pdf
+        cal_work_dir = report_to.with_suffix('')
     else:
-        cal_work_dir = out_path / workdir_name
+        cal_work_dir = out_path / unique_name
     cal_work_dir.mkdir(parents=True)
 
     # Write all input parameters to rst file to be included to final report
@@ -660,7 +659,7 @@ def run(argv=None):
         parm_subdict[name] = p.value
 
     metadata["pycalibration-version"] = version
-    metadata["report-path"] = f"{report_to}.pdf" if report_to \
+    metadata["report-path"] = str(report_to) if report_to \
         else '# REPORT SKIPPED #'
     metadata['reproducible'] = not args['not_reproducible']
     metadata["concurrency"] = {
@@ -689,11 +688,6 @@ def run(argv=None):
 
     folder = get_par_attr(parms, 'in_folder', 'value', '')
 
-    if args["request_time"] == "Now":
-        request_time = datetime.now().strftime('%Y-%m-%dT%H:%M:%S')
-    else:
-        request_time = args["request_time"]
-
     pre_jobs = []
     cluster_cores = concurrency.get("cluster cores", 8)
     # Check if there are pre-notebooks
@@ -823,7 +817,7 @@ def run(argv=None):
                 'author': author,
                 'report_to': report_to,
                 'in_folder': folder,
-                'request_time': request_time,
+                'request_time': request_time.strftime("%Y-%m-%dT%H:%M:%S"),
                 'submission_time': submission_time,
                 }
 
diff --git a/src/xfel_calibrate/finalize.py b/src/xfel_calibrate/finalize.py
index 0df50980e32551a0db3f9b1bf3a8d6548856af28..41b3f9dd4b84f61a666647064959b966ba915543 100644
--- a/src/xfel_calibrate/finalize.py
+++ b/src/xfel_calibrate/finalize.py
@@ -414,7 +414,7 @@ def finalize(joblist, finaljob, cal_work_dir, out_path, version, title, author,
             version,
             report_to,
         )
-        det = metadata['calibration-configurations'].get('karabo-id', report_to.name)
+        det = metadata['calibration-configurations'].get('karabo-id', report_to.stem)
     else:
         try:
             det = metadata['calibration-configurations']['karabo-id']
diff --git a/src/xfel_calibrate/nb_args.py b/src/xfel_calibrate/nb_args.py
index e3e1dc5e2adbdd040bb566790064672321a52bd5..93c6eba71ed8b0df2bbb22bcca83d2652a7434d0 100644
--- a/src/xfel_calibrate/nb_args.py
+++ b/src/xfel_calibrate/nb_args.py
@@ -53,8 +53,7 @@ def make_initial_parser(**kwargs):
                         help="Prepare notebooks but don't run them")
 
     parser.add_argument('--report-to', type=str,
-                        help='Filename (and optionally path) for output'
-                             ' report')
+                        help='Full path for the PDF report output')
 
     parser.add_argument('--not-reproducible', action='store_true',
                         help='Disable checks to allow the processing result '
diff --git a/src/xfel_calibrate/repeat.py b/src/xfel_calibrate/repeat.py
index e8d94134f3b8a46384f1272b754fea66c070b80c..3112a960f5f428cce261595347f36c10bdc5739f 100644
--- a/src/xfel_calibrate/repeat.py
+++ b/src/xfel_calibrate/repeat.py
@@ -13,7 +13,6 @@ from cal_tools.tools import CalibrationMetadata
 from .calibrate import (
     JobChain, SlurmOptions, run_finalize, get_pycalib_version,
 )
-from .settings import temp_path
 
 # This function is copied and modified from Python 3.8.10
 # Copyright © 2001-2022 Python Software Foundation; All Rights Reserved
@@ -123,30 +122,35 @@ def main(argv=None):
     start_time = datetime.now()
     run_uuid = f"t{start_time:%y%m%d_%H%M%S}"
 
-    cal_work_dir = Path(temp_path, f'slurm_out_repeat_{run_uuid}')
-    copytree_no_metadata(
-        args.from_dir, cal_work_dir, ignore=shutil.ignore_patterns('slurm-*.out')
-    )
-    print(f"New working directory: {cal_work_dir}")
-
-    cal_metadata = CalibrationMetadata(cal_work_dir)
-    parameters = cal_metadata['calibration-configurations']
+    parameters = CalibrationMetadata(args.from_dir)['calibration-configurations']
+    karabo_id = parameters['karabo-id']
 
     out_folder = parameters['out-folder']
     params_to_set = {'metadata_folder': "."}
     if args.out_folder:
         out_folder = parameters['out-folder'] = os.path.abspath(args.out_folder)
         params_to_set['out_folder'] = out_folder
-    update_notebooks_params(cal_work_dir, params_to_set)
 
     if args.report_to:
-        report_to = os.path.abspath(args.report_to)
+        report_to = Path(args.report_to).with_suffix('.pdf').absolute()
     else:  # Default to saving report in output folder
-        report_to = str(Path(out_folder, f'xfel-calibrate-repeat-{run_uuid}'))
-    cal_metadata['report-path'] = f'{report_to}.pdf'
+        report_to = Path(out_folder, f'{karabo_id}-repeat-{run_uuid}.pdf')
 
+    # Copy working directory to new location
+    cal_work_dir = report_to.with_suffix('')
+    copytree_no_metadata(
+        args.from_dir, cal_work_dir, ignore=shutil.ignore_patterns('slurm-*.out')
+    )
+    print(f"New working directory: {cal_work_dir}")
+
+    # Update metadata YAML file & notebooks with any changes
+    cal_metadata = CalibrationMetadata(cal_work_dir)
+    cal_metadata['calibration-configurations'] = parameters
+    cal_metadata['report-path'] = str(report_to)
     cal_metadata.save()
 
+    update_notebooks_params(cal_work_dir, params_to_set)
+
     # finalize & some notebooks expect yaml metadata in the output folder
     Path(out_folder).mkdir(parents=True, exist_ok=True)
     shutil.copy(cal_work_dir / 'calibration_metadata.yml', out_folder)
diff --git a/tests/test_xfel_calibrate/conftest.py b/tests/test_xfel_calibrate/conftest.py
index a98c410259385d6dc4a7d837b374331681635996..21161c99edb124c607acca3de6a60c10a4fda88e 100644
--- a/tests/test_xfel_calibrate/conftest.py
+++ b/tests/test_xfel_calibrate/conftest.py
@@ -215,7 +215,9 @@ class CalibrateCall:
         self.in_folder = in_folder
         self.out_folder = out_folder
 
-        self.args = [command, detector, cal_type, '--report-to', str(reports_dir)]
+        self.args = [
+            command, detector, cal_type, '--report-to', str(reports_dir / 'test.pdf')
+        ]
         if in_folder:
             self.args.extend(["--in-folder", str(self.in_folder)])
         if out_folder:
diff --git a/webservice/webservice.py b/webservice/webservice.py
index 9fa488f2308fe1d25ccb9c908b72d545822ed796..34da59a3d9f68f511f78e5383ca263ed67041fd1 100644
--- a/webservice/webservice.py
+++ b/webservice/webservice.py
@@ -956,7 +956,7 @@ class ActionsServer:
         This will trigger a correction process to be launched for that run in
         the given cycle and proposal.
         """
-        request_time = datetime.now().strftime('%Y-%m-%dT%H:%M:%S')
+        request_time = datetime.now()
         try:
             runnr = runnr.strip('r')
 
@@ -970,9 +970,9 @@ class ActionsServer:
                 return msg.encode()
 
             with time_db_transaction(self.job_db, 'Insert request'):
-                cur = self.job_db.execute(  # 2
+                cur = self.job_db.execute(
                     "INSERT INTO requests VALUES (NULL, ?, ?, ?, 'CORRECT', ?)",
-                    (rid, proposal, int(runnr), request_time)
+                    (rid, proposal, int(runnr), request_time.isoformat())
                 )
                 req_id = cur.lastrowid
 
@@ -1105,7 +1105,7 @@ class ActionsServer:
             with time_db_transaction(self.job_db, 'Insert request'):
                 cur = self.job_db.execute(
                     "INSERT INTO requests VALUES (NULL, ?, ?, ?, 'CORRECT', ?)",
-                    (rid, proposal, int(runnr), request_time.strftime('%Y-%m-%dT%H:%M:%S'))
+                    (rid, proposal, int(runnr), request_time.isoformat())
                 )
                 req_id = cur.lastrowid
 
@@ -1170,7 +1170,7 @@ class ActionsServer:
                     '--env-cache',
                     f'/gpfs/exfel/data/scratch/{getuser()}/calib-repeat-envs',
                     '--report-to',
-                    f'{reports_dir}/{karabo_id}_RECORRECT_{request_time:%y%m%d_%H%M%S}'
+                    f'{reports_dir}/{karabo_id}_RECORRECT_{request_time:%y%m%d_%H%M%S_%f}'
                 ]
 
                 with time_db_transaction(self.job_db, 'Insert execution'):
@@ -1216,7 +1216,7 @@ class ActionsServer:
         :param runnr: is the run number in integer form, i.e. without leading
                      "r"
         """
-        request_time = datetime.now().strftime('%Y-%m-%dT%H:%M:%S')
+        request_time = datetime.now()
         try:
             pdus, karabo_das, wait_runs = ast.literal_eval(','.join(extra))
 
@@ -1236,7 +1236,7 @@ class ActionsServer:
             with time_db_transaction(self.job_db, 'Insert request'):
                 cur = self.job_db.execute(
                     "INSERT INTO requests VALUES (NULL, ?, ?, ?, 'DARK', ?)",
-                    (rid, proposal, int(wait_runs[-1]), request_time)
+                    (rid, proposal, int(wait_runs[-1]), request_time.isoformat())
                 )
                 req_id = cur.lastrowid
 
@@ -1445,9 +1445,9 @@ class ActionsServer:
                 action=action, instrument=instrument,
                 cycle=cycle, proposal=proposal,
                 runs="_".join([f"r{r}" for r in run_nrs]),
-                time_stamp=datetime.now().strftime('%y%m%d_%H%M%S'),
+                time_stamp=request_time.strftime('%y%m%d_%H%M%S_%f'),
                 det_instance=karabo_id,
-                request_time=request_time
+                request_time=request_time.isoformat(),
             ).split()
             cmd = parse_config(cmd, dconfig)