# Copyright (C) 2024 qBraid
#
# This file is part of the qBraid-SDK
#
# The qBraid-SDK is free software released under the GNU General Public License v3
# or later. You can redistribute and/or modify it under the terms of the GPL v3.
# See the LICENSE file in the project root or <https://www.gnu.org/licenses/gpl-3.0.html>.
#
# THERE IS NO WARRANTY for the qBraid-SDK, as per Section 15 of the GPL v3.
"""
Module for managing and retrieving custom program type aliases
"""
from __future__ import annotations
from typing import TYPE_CHECKING, Optional, Type
from .exceptions import ProgramTypeError
from .exceptions import QasmError as QbraidQasmError
from .registry import QPROGRAM_REGISTRY, QPROGRAM_TYPES
from .typer import IonQDict, get_qasm_type_alias
if TYPE_CHECKING:
import qbraid.programs
def find_str_type_alias(registry: dict[str, Type] = QPROGRAM_REGISTRY) -> Optional[str]:
"""Find additional keys with type 'str' in the registry."""
str_keys = [k for k, v in registry.items() if v is str and k not in ("qasm2", "qasm3")]
if len(str_keys) == 0:
return None
if len(str_keys) == 1:
return str_keys[0]
raise ValueError(f"Multiple additional keys with type 'str' found: {str_keys}")
def _get_program_type_alias(program: qbraid.programs.QPROGRAM) -> str:
"""
Get the type alias of a quantum program from registry.
Args:
program (qbraid.programs.QPROGRAM): The quantum program to get the type of.
Returns:
str: The type of the quantum program.
Raises:
ProgramTypeError: If the program type does not match any registered program types.
"""
if isinstance(program, type):
raise ProgramTypeError(message="Expected an instance of a quantum program, not a type.")
if isinstance(program, str):
try:
return get_qasm_type_alias(program)
except QbraidQasmError as err:
package = find_str_type_alias()
if package is not None:
return package
raise ProgramTypeError(
message=(
"Input of program of type string does not represent a valid OpenQASM program, "
"and no additional string type aliases are registered."
)
) from err
if isinstance(program, IonQDict):
return IonQDict.__alias__
matched = []
for alias, program_type in QPROGRAM_REGISTRY.items():
if isinstance(program, (program_type, type(program_type))):
matched.append(alias)
if len(matched) == 1:
return matched[0]
if len(matched) > 1:
raise ProgramTypeError(
message=(
f"Program of type '{type(program)}' matches multiple registered program types: "
f"{matched}."
)
)
raise ProgramTypeError(
message=(
f"Program of type '{type(program)}' does not match any registered "
f"program types. Registered program types are: {QPROGRAM_TYPES}."
)
)
[docs]
def get_program_type_alias(program: qbraid.programs.QPROGRAM, safe: bool = False) -> Optional[str]:
"""
Get the type alias of a quantum program from registry.
Args:
program (qbraid.programs.QPROGRAM): The quantum program to get the type of.
safe (bool): If True, return None if the program type does not match any registered
program types. Defaults to False.
Returns:
str: The type of the quantum program.
None: If the program type does not match any registered program types and safe is True.
Raises:
ProgramTypeError: If the program type does not match any registered program types.
"""
try:
return _get_program_type_alias(program)
except ProgramTypeError as err:
if safe:
return None
raise err