|
1 | 1 | # Copyright (c) Microsoft Corporation.
|
2 | 2 | # Licensed under the MIT License.
|
3 | 3 |
|
4 |
| -from typing import Any, Dict, Optional, Union, List |
| 4 | +from typing import Any, Callable, Dict, Optional, TypedDict, Union, List |
5 | 5 | from ._native import Interpreter, Output, TargetProfile, StateDump
|
6 | 6 | from .estimator._estimator import EstimatorResult, EstimatorParams
|
7 | 7 | import json
|
@@ -61,6 +61,7 @@ def get_interpreter() -> Interpreter:
|
61 | 61 | global _interpreter
|
62 | 62 | if _interpreter is None:
|
63 | 63 | init()
|
| 64 | + assert _interpreter is not None, "Failed to initialize the Q# interpreter." |
64 | 65 | return _interpreter
|
65 | 66 |
|
66 | 67 |
|
@@ -93,23 +94,59 @@ def eval_file(path: str) -> Any:
|
93 | 94 | return eval(f.read())
|
94 | 95 |
|
95 | 96 |
|
96 |
| -def run(entry_expr: str, shots: int) -> Any: |
| 97 | +class ShotResult(TypedDict): |
| 98 | + """ |
| 99 | + A single result of a shot. |
| 100 | + """ |
| 101 | + |
| 102 | + events: List[Output] |
| 103 | + result: Any |
| 104 | + |
| 105 | + |
| 106 | +def run( |
| 107 | + entry_expr: str, |
| 108 | + shots: int, |
| 109 | + *, |
| 110 | + on_result: Optional[Callable[[ShotResult], None]] = None, |
| 111 | + save_events: bool = False, |
| 112 | +) -> List[Any]: |
97 | 113 | """
|
98 | 114 | Runs the given Q# expression for the given number of shots.
|
99 | 115 | Each shot uses an independent instance of the simulator.
|
100 | 116 |
|
101 | 117 | :param entry_expr: The entry expression.
|
102 | 118 | :param shots: The number of shots to run.
|
| 119 | + :param on_result: A callback function that will be called with each result. |
| 120 | + :param save_events: If true, the output of each shot will be saved. If false, they will be printed. |
103 | 121 |
|
104 |
| - :returns values: A list of results or runtime errors. |
| 122 | + :returns values: A list of results or runtime errors. If `save_events` is true, |
| 123 | + a List of ShotResults is returned. |
105 | 124 |
|
106 | 125 | :raises QSharpError: If there is an error interpreting the input.
|
107 | 126 | """
|
108 | 127 |
|
109 |
| - def callback(output: Output) -> None: |
| 128 | + results: List[ShotResult] = [] |
| 129 | + |
| 130 | + def print_output(output: Output) -> None: |
110 | 131 | print(output)
|
111 | 132 |
|
112 |
| - return get_interpreter().run(entry_expr, shots, callback) |
| 133 | + def on_save_events(output: Output) -> None: |
| 134 | + # Append the output to the last shot's output list |
| 135 | + results[-1]["events"].append(output) |
| 136 | + |
| 137 | + for _ in range(shots): |
| 138 | + results.append({"result": None, "events": []}) |
| 139 | + run_results = get_interpreter().run( |
| 140 | + entry_expr, 1, on_save_events if save_events else print_output |
| 141 | + ) |
| 142 | + results[-1]["result"] = run_results[0] |
| 143 | + if on_result: |
| 144 | + on_result(results[-1]) |
| 145 | + |
| 146 | + if save_events: |
| 147 | + return results |
| 148 | + else: |
| 149 | + return [shot["result"] for shot in results] |
113 | 150 |
|
114 | 151 |
|
115 | 152 | # Class that wraps generated QIR, which can be used by
|
|
0 commit comments