Skip to main contentIBM Quantum Documentation

Save and retrieve jobs

Package versions

The code on this page was developed using the following requirements. We recommend using these versions or newer.

qiskit[all]~=1.3.1
qiskit-ibm-runtime~=0.34.0
qiskit-aer~=0.15.1
qiskit-serverless~=0.18.1
qiskit-ibm-catalog~=0.2
qiskit-addon-sqd~=0.8.1
qiskit-addon-utils~=0.1.0
qiskit-addon-mpf~=0.2.0
scipy~=1.14.1
qiskit-addon-aqc-tensor~=0.1.2
qiskit-addon-obp~=0.1.0
scipy~=1.14.1
pyscf~=2.7.0

Quantum workflows often take a while to complete and can run over many sessions. Restarting your Python kernel means you'll lose any results stored in memory. To avoid loss of data, you can save results to file and retrieve results of past jobs from IBM Quantum™ so your next session can continue where you left off.


Retrieve jobs from IBM Quantum

IBM Quantum automatically stores results from every job for you to retrieve at a later date. Use this feature to continue quantum programs across kernel restarts and review past results. You can get the ID of a job programmatically through its job_id method, or you can see all your submitted jobs and their IDs through the Workloads dashboard.

To find a job programatically, use the QiskitRuntimeService.jobs method. By default, this returns the most recent jobs, but you can also filter jobs by backend name, creation date, and more. The following cell finds any jobs submitted in the last three months. The created_after argument must be a datetime.datetime object.

import datetime
from qiskit_ibm_runtime import QiskitRuntimeService
 
three_months_ago = datetime.datetime.now() - datetime.timedelta(days=90)
 
service = QiskitRuntimeService()
jobs_in_last_three_months = service.jobs(created_after=three_months_ago)
jobs_in_last_three_months[:3]  # show first three jobs

Output:

[<RuntimeJob('cxg030c6t010008cme5g', 'estimator')>,
 <RuntimeJob('cxg02zvvw7kg008s3vk0', 'estimator')>,
 <RuntimeJob('cxg02z36t010008cme50', 'estimator')>]

You can also select by backend, job state, session, and more. For more information, see QiskitRuntimeService.jobs in the API documentation.

Once you have the job ID, use the QiskitRuntimeService.job method to retrieve it.

# Get ID of most recent successful job for demonstration.
# This will not work if you've never successfully run a job.
successful_job = next(
    j for j in service.jobs(limit=1000) if j.status().name == "DONE"
)
job_id = successful_job.job_id()
print(job_id)

Output:

/tmp/ipykernel_5348/3090032923.py:4: DeprecationWarning: In a future release of qiskit-ibm-runtime no sooner than 3 months after the release date of 0.30.0, RuntimeJob.status() will be returned as a string instead of an instance of `JobStatus`. To prepare for this change, you can use the idiom `status.name if isinstance(status, JobStatus) else status`.
  j for j in service.jobs(limit=1000) if j.status().name == "DONE"
cxg030c6t010008cme5g
retrieved_job = service.job(job_id)
retrieved_job.result()

Output:

PrimitiveResult([PubResult(data=DataBin(evs=np.ndarray(<shape=(6,), dtype=float64>), stds=np.ndarray(<shape=(6,), dtype=float64>), evs_noise_factors=np.ndarray(<shape=(6, 2), dtype=float64>), stds_noise_factors=np.ndarray(<shape=(6, 2), dtype=float64>), ensemble_stds_noise_factors=np.ndarray(<shape=(6, 2), dtype=float64>), evs_extrapolated=np.ndarray(<shape=(6, 2, 3), dtype=float64>), stds_extrapolated=np.ndarray(<shape=(6, 2, 3), dtype=float64>), shape=(6,)), metadata={'shots': 100000, 'target_precision': 0.003162277660168379, 'circuit_metadata': {}, 'resilience': {'zne': {'extrapolator': array(['exponential', 'exponential', 'exponential', 'exponential',
       'exponential', 'exponential'], dtype='<U11')}, 'layer_noise': {'noise_overhead': 2.3541035574729365, 'total_mitigated_layers': 6, 'unique_mitigated_layers': 2, 'unique_mitigated_layers_noise_overhead': [1.1356441238948969, 1.1896748089636464]}}, 'num_randomizations': 1000})], metadata={'dynamical_decoupling': {'enable': False, 'sequence_type': 'XX', 'extra_slack_distribution': 'middle', 'scheduling_method': 'alap'}, 'twirling': {'enable_gates': True, 'enable_measure': True, 'num_randomizations': 'auto', 'shots_per_randomization': 100, 'interleave_randomizations': True, 'strategy': 'active-accum'}, 'resilience': {'measure_mitigation': True, 'zne_mitigation': True, 'pec_mitigation': False, 'zne': {'noise_factors': [1, 4], 'extrapolator': ['exponential', 'linear'], 'extrapolated_noise_factors': [0, 1, 4]}, 'layer_noise_model': [LayerError(circuit=<qiskit.circuit.quantumcircuit.QuantumCircuit object at 0x7fbb6b9ac850>, qubits=[0, 1, 2, 3, 4, 5], error=PauliLindbladError(generators=['IIIIIX', 'IIIIIY', 'IIIIIZ', 'IIIIXI', 'IIIIXX', 'IIIIXY', 'IIIIXZ',
 'IIIIYI', 'IIIIYX', 'IIIIYY', 'IIIIYZ', 'IIIIZI', 'IIIIZX', 'IIIIZY',
 'IIIIZZ', 'IIIXII', 'IIIXXI', 'IIIXYI', 'IIIXZI', 'IIIYII', 'IIIYXI',
 'IIIYYI', 'IIIYZI', 'IIIZII', 'IIIZXI', 'IIIZYI', 'IIIZZI', 'IIXIII',
 'IIXXII', 'IIXYII', 'IIXZII', 'IIYIII', 'IIYXII', 'IIYYII', 'IIYZII',
 'IIZIII', 'IIZXII', 'IIZYII', 'IIZZII', 'IXIIII', 'IXXIII', 'IXYIII',
 'IXZIII', 'IYIIII', 'IYXIII', 'IYYIII', 'IYZIII', 'IZIIII', 'IZXIII',
 'IZYIII', 'IZZIII', 'XIIIII', 'XXIIII', 'XYIIII', 'XZIIII', 'YIIIII',
 'YXIIII', 'YYIIII', 'YZIIII', 'ZIIIII', 'ZXIIII', 'ZYIIII', 'ZZIIII'], rates=[0.00155, 0.00046, 0.00069, 0.00014, 0.0005, 0.0, 0.00016, 0.00031, 0.00033, 0.00016, 6e-05, 8e-05, 0.00017, 0.00069, 0.00046, 0.00102, 0.0, 0.0, 2e-05, 0.00065, 8e-05, 0.00011, 0.00031, 0.00225, 0.00011, 8e-05, 0.00031, 0.00052, 0.0011, 0.0, 0.00014, 0.00066, 0.00096, 0.00012, 0.0, 0.00058, 0.00012, 0.00275, 0.00115, 0.00015, 0.00012, 5e-05, 0.00019, 0.00033, 5e-05, 0.00012, 0.00019, 0.00067, 0.00027, 0.00027, 0.00311, 0.00352, 0.00069, 0.00051, 7e-05, 0.00036, 0.00029, 0.0, 0.00073, 0.00073, 0.0, 0.00022, 0.00036]))), LayerError(circuit=<qiskit.circuit.quantumcircuit.QuantumCircuit object at 0x7fbb6b9af610>, qubits=[0, 1, 2, 3, 4, 5], error=PauliLindbladError(generators=['IIIIIX', 'IIIIIY', 'IIIIIZ', 'IIIIXI', 'IIIIXX', 'IIIIXY', 'IIIIXZ',
 'IIIIYI', 'IIIIYX', 'IIIIYY', 'IIIIYZ', 'IIIIZI', 'IIIIZX', 'IIIIZY',
 'IIIIZZ', 'IIIXII', 'IIIXXI', 'IIIXYI', 'IIIXZI', 'IIIYII', 'IIIYXI',
 'IIIYYI', 'IIIYZI', 'IIIZII', 'IIIZXI', 'IIIZYI', 'IIIZZI', 'IIXIII',
 'IIXXII', 'IIXYII', 'IIXZII', 'IIYIII', 'IIYXII', 'IIYYII', 'IIYZII',
 'IIZIII', 'IIZXII', 'IIZYII', 'IIZZII', 'IXIIII', 'IXXIII', 'IXYIII',
 'IXZIII', 'IYIIII', 'IYXIII', 'IYYIII', 'IYZIII', 'IZIIII', 'IZXIII',
 'IZYIII', 'IZZIII', 'XIIIII', 'XXIIII', 'XYIIII', 'XZIIII', 'YIIIII',
 'YXIIII', 'YYIIII', 'YZIIII', 'ZIIIII', 'ZXIIII', 'ZYIIII', 'ZZIIII'], rates=[0.00115, 0.00115, 0.00824, 0.00093, 0.0, 0.00021, 2e-05, 0.0, 0.00024, 0.00016, 0.0, 0.00037, 0.00025, 0.00017, 0.0, 0.00071, 0.00119, 0.00017, 0.00028, 0.00077, 0.00112, 6e-05, 0.00023, 0.00192, 0.00033, 0.00078, 0.0004, 0.00185, 8e-05, 8e-05, 0.00056, 0.00021, 0.00013, 0.00026, 0.00042, 0.0, 0.00021, 8e-05, 0.00037, 0.00216, 0.0009, 5e-05, 0.0, 0.00089, 0.00217, 0.00027, 0.00014, 0.00149, 0.00042, 0.00066, 0.00096, 0.00285, 0.0, 0.0, 0.0, 0.00234, 0.0, 0.0, 0.0, 0.00292, 1e-05, 1e-05, 8e-05])))]}, 'version': 2})
Qiskit Code Assistant

Always forgetting how to retrieve a job? Try this prompt with Qiskit Code Assistant:

# Retrieve job JOB_ID from IBM Runtime and print the result

New to the code assistant? See Qiskit Code Assistant for installation and usage. Note this is an experimental feature and only available to IBM Quantum™ Premium Plan users.


Save results to disk

You may also want to save results to disk. To do this, use Python's built-in JSON library with Qiskit Runtime's encoders.

import json
from qiskit_ibm_runtime import RuntimeEncoder
 
with open("result.json", "w") as file:
    json.dump(retrieved_job.result(), file, cls=RuntimeEncoder)

You can then load this array from disk in a separate kernel.

from qiskit_ibm_runtime import RuntimeDecoder
 
with open("result.json", "r") as file:
    result = json.load(file, cls=RuntimeDecoder)
 
result

Output:

PrimitiveResult([PubResult(data=DataBin(evs=np.ndarray(<shape=(6,), dtype=float64>), stds=np.ndarray(<shape=(6,), dtype=float64>), evs_noise_factors=np.ndarray(<shape=(6, 2), dtype=float64>), stds_noise_factors=np.ndarray(<shape=(6, 2), dtype=float64>), ensemble_stds_noise_factors=np.ndarray(<shape=(6, 2), dtype=float64>), evs_extrapolated=np.ndarray(<shape=(6, 2, 3), dtype=float64>), stds_extrapolated=np.ndarray(<shape=(6, 2, 3), dtype=float64>), shape=(6,)), metadata={'shots': 100000, 'target_precision': 0.003162277660168379, 'circuit_metadata': {}, 'resilience': {'zne': {'extrapolator': array(['exponential', 'exponential', 'exponential', 'exponential',
       'exponential', 'exponential'], dtype='<U11')}, 'layer_noise': {'noise_overhead': 2.3541035574729365, 'total_mitigated_layers': 6, 'unique_mitigated_layers': 2, 'unique_mitigated_layers_noise_overhead': [1.1356441238948969, 1.1896748089636464]}}, 'num_randomizations': 1000})], metadata={'dynamical_decoupling': {'enable': False, 'sequence_type': 'XX', 'extra_slack_distribution': 'middle', 'scheduling_method': 'alap'}, 'twirling': {'enable_gates': True, 'enable_measure': True, 'num_randomizations': 'auto', 'shots_per_randomization': 100, 'interleave_randomizations': True, 'strategy': 'active-accum'}, 'resilience': {'measure_mitigation': True, 'zne_mitigation': True, 'pec_mitigation': False, 'zne': {'noise_factors': [1, 4], 'extrapolator': ['exponential', 'linear'], 'extrapolated_noise_factors': [0, 1, 4]}, 'layer_noise_model': [LayerError(circuit=<qiskit.circuit.quantumcircuit.QuantumCircuit object at 0x7fbbc463b750>, qubits=[0, 1, 2, 3, 4, 5], error=PauliLindbladError(generators=['IIIIIX', 'IIIIIY', 'IIIIIZ', 'IIIIXI', 'IIIIXX', 'IIIIXY', 'IIIIXZ',
 'IIIIYI', 'IIIIYX', 'IIIIYY', 'IIIIYZ', 'IIIIZI', 'IIIIZX', 'IIIIZY',
 'IIIIZZ', 'IIIXII', 'IIIXXI', 'IIIXYI', 'IIIXZI', 'IIIYII', 'IIIYXI',
 'IIIYYI', 'IIIYZI', 'IIIZII', 'IIIZXI', 'IIIZYI', 'IIIZZI', 'IIXIII',
 'IIXXII', 'IIXYII', 'IIXZII', 'IIYIII', 'IIYXII', 'IIYYII', 'IIYZII',
 'IIZIII', 'IIZXII', 'IIZYII', 'IIZZII', 'IXIIII', 'IXXIII', 'IXYIII',
 'IXZIII', 'IYIIII', 'IYXIII', 'IYYIII', 'IYZIII', 'IZIIII', 'IZXIII',
 'IZYIII', 'IZZIII', 'XIIIII', 'XXIIII', 'XYIIII', 'XZIIII', 'YIIIII',
 'YXIIII', 'YYIIII', 'YZIIII', 'ZIIIII', 'ZXIIII', 'ZYIIII', 'ZZIIII'], rates=[0.00155, 0.00046, 0.00069, 0.00014, 0.0005, 0.0, 0.00016, 0.00031, 0.00033, 0.00016, 6e-05, 8e-05, 0.00017, 0.00069, 0.00046, 0.00102, 0.0, 0.0, 2e-05, 0.00065, 8e-05, 0.00011, 0.00031, 0.00225, 0.00011, 8e-05, 0.00031, 0.00052, 0.0011, 0.0, 0.00014, 0.00066, 0.00096, 0.00012, 0.0, 0.00058, 0.00012, 0.00275, 0.00115, 0.00015, 0.00012, 5e-05, 0.00019, 0.00033, 5e-05, 0.00012, 0.00019, 0.00067, 0.00027, 0.00027, 0.00311, 0.00352, 0.00069, 0.00051, 7e-05, 0.00036, 0.00029, 0.0, 0.00073, 0.00073, 0.0, 0.00022, 0.00036]))), LayerError(circuit=<qiskit.circuit.quantumcircuit.QuantumCircuit object at 0x7fbb6baf4390>, qubits=[0, 1, 2, 3, 4, 5], error=PauliLindbladError(generators=['IIIIIX', 'IIIIIY', 'IIIIIZ', 'IIIIXI', 'IIIIXX', 'IIIIXY', 'IIIIXZ',
 'IIIIYI', 'IIIIYX', 'IIIIYY', 'IIIIYZ', 'IIIIZI', 'IIIIZX', 'IIIIZY',
 'IIIIZZ', 'IIIXII', 'IIIXXI', 'IIIXYI', 'IIIXZI', 'IIIYII', 'IIIYXI',
 'IIIYYI', 'IIIYZI', 'IIIZII', 'IIIZXI', 'IIIZYI', 'IIIZZI', 'IIXIII',
 'IIXXII', 'IIXYII', 'IIXZII', 'IIYIII', 'IIYXII', 'IIYYII', 'IIYZII',
 'IIZIII', 'IIZXII', 'IIZYII', 'IIZZII', 'IXIIII', 'IXXIII', 'IXYIII',
 'IXZIII', 'IYIIII', 'IYXIII', 'IYYIII', 'IYZIII', 'IZIIII', 'IZXIII',
 'IZYIII', 'IZZIII', 'XIIIII', 'XXIIII', 'XYIIII', 'XZIIII', 'YIIIII',
 'YXIIII', 'YYIIII', 'YZIIII', 'ZIIIII', 'ZXIIII', 'ZYIIII', 'ZZIIII'], rates=[0.00115, 0.00115, 0.00824, 0.00093, 0.0, 0.00021, 2e-05, 0.0, 0.00024, 0.00016, 0.0, 0.00037, 0.00025, 0.00017, 0.0, 0.00071, 0.00119, 0.00017, 0.00028, 0.00077, 0.00112, 6e-05, 0.00023, 0.00192, 0.00033, 0.00078, 0.0004, 0.00185, 8e-05, 8e-05, 0.00056, 0.00021, 0.00013, 0.00026, 0.00042, 0.0, 0.00021, 8e-05, 0.00037, 0.00216, 0.0009, 5e-05, 0.0, 0.00089, 0.00217, 0.00027, 0.00014, 0.00149, 0.00042, 0.00066, 0.00096, 0.00285, 0.0, 0.0, 0.0, 0.00234, 0.0, 0.0, 0.0, 0.00292, 1e-05, 1e-05, 8e-05])))]}, 'version': 2})
Was this page helpful?
Report a bug or request content on GitHub.