# 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 registering custom program types and aliases
"""
import sys
from typing import Any, Optional, Type, TypeVar, Union
from qbraid._entrypoints import get_entrypoints
from ._import import _QPROGRAM_ALIASES, _QPROGRAM_REGISTRY, _QPROGRAM_TYPES, NATIVE_REGISTRY
from .experiment import ExperimentType
from .typer import QbraidMetaType
QPROGRAM_NATIVE = Union[tuple(_QPROGRAM_TYPES)]
QPROGRAM_REGISTRY = _QPROGRAM_REGISTRY
QPROGRAM_ALIASES = _QPROGRAM_ALIASES
QPROGRAM_TYPES = _QPROGRAM_TYPES
QPROGRAM = TypeVar("QPROGRAM", bound=Any)
def derive_program_type_alias(program_type: Type[Any], use_submodule: bool = False) -> str:
"""
Determines an alias for the given program type based on its module or class name.
Args:
program_type (Type[Any]): The class or type for which to determine the alias.
use_submodule (bool): If True, alias will be based on first two parts of module name.
Returns:
str: The determined alias for the program type.
Raises:
ValueError: If the alias cannot be determined, or if use_submodule is True and the
module name is invalid or belongs to __main__ or builtins.
"""
if isinstance(program_type, QbraidMetaType):
return program_type.__alias__
try:
module_name = program_type.__module__
module_parts = module_name.split(".")
module_root = module_parts[0]
if module_root in {"__main__", "builtins"}:
if use_submodule:
raise ValueError("Cannot use submodule aliasing with __main__ or builtins module.")
alias = program_type.__name__
else:
if use_submodule:
if len(module_parts) < 2:
raise ValueError(
"Module name does not have at least two parts for submodule aliasing."
)
alias = f"{module_root}_{module_parts[1]}"
else:
alias = module_root
return alias.lower()
except (AttributeError, IndexError, TypeError) as err:
raise ValueError(
"Failed to automatically determine an alias from the program type's module."
) from err
[docs]
def register_program_type(
program_type: Type[Any], alias: Optional[str] = None, overwrite: bool = False
) -> None:
"""
Registers a user-defined program type under the specified alias.
Args:
program_type (Type[Any]): The actual Python type or a callable that returns a type.
This can be a built-in type like str, a class, or any callable.
alias (optional, str): The alias to register the program type under.
overwrite (optional, bool): Whether to overwrite an existing alias with the new type.
Raises:
ValueError: If the alias is already registered with a different type,
if the program type is already registered under a different alias,
or if trying to register more than one additional 'str' type beyond
'qasm2' and 'qasm3'.
"""
if not alias:
alias = derive_program_type_alias(program_type)
normalized_alias = alias.lower()
# Check if the alias is already used and if it maps to a different type
if normalized_alias in QPROGRAM_REGISTRY:
registered_type = QPROGRAM_REGISTRY[normalized_alias]
if registered_type != program_type and overwrite is False:
if (
isinstance(registered_type, QbraidMetaType)
and program_type == registered_type.__bound__
):
program_type = registered_type
else:
raise ValueError(f"Alias '{alias}' is already registered with a different type.")
# Check if the type is already registered under any other alias
existing_alias = next((k for k, v in QPROGRAM_REGISTRY.items() if v == program_type), None)
if existing_alias and existing_alias != normalized_alias and overwrite is False:
if program_type is str:
str_types = [
k for k, v in QPROGRAM_REGISTRY.items() if v is str and k not in ("qasm2", "qasm3")
]
if (
len(str_types) >= 1
and normalized_alias not in str_types
and normalized_alias not in ("qasm2", "qasm3")
):
raise ValueError(
"Cannot register more than one additional 'str' type beyond 'qasm2', 'qasm3'."
)
else:
raise ValueError(
f"Program type '{program_type}' already registered under alias '{existing_alias}'."
)
# Register the new type and alias
QPROGRAM_REGISTRY[normalized_alias] = program_type
QPROGRAM_ALIASES.add(normalized_alias)
QPROGRAM_TYPES.add(program_type)
[docs]
def unregister_program_type(alias: str, raise_error: bool = True) -> None:
"""
Unregisters the program type associated with the given alias.
Args:
alias (str): The alias to unregister.
raise_error (bool): If True, raises KeyError if the alias is not found. Defaults to True.
Raises:
KeyError: If the alias is not registered and raise_error is True.
"""
normalized_alias = alias.lower()
QPROGRAM_ALIASES.discard(normalized_alias)
if normalized_alias not in QPROGRAM_REGISTRY:
if raise_error:
raise KeyError(f"No program type registered under the alias '{alias}'.")
return
program_type = QPROGRAM_REGISTRY.pop(normalized_alias)
if not any(pt == program_type for pt in QPROGRAM_REGISTRY.values()):
QPROGRAM_TYPES.discard(program_type)
def is_registered_alias_native(alias: str) -> bool:
"""
Determine if the registered program type for a given alias matches the native program type.
Args:
alias (str): The alias to check against the native and registered program types.
Returns:
bool: True if the alias is registered and its program type matches the native type,
otherwise False.
"""
native_type = NATIVE_REGISTRY.get(alias)
registered_type = QPROGRAM_REGISTRY.get(alias)
return native_type is not None and native_type == registered_type
def get_native_experiment_type(native_alias: str) -> ExperimentType:
"""Returns the ExperimentType corresponding to the native program type alias."""
module = "programs"
group = f"qbraid.{module}"
native_no_ep = {
"openqasm3": ExperimentType.GATE_MODEL,
"pyqir": ExperimentType.GATE_MODEL,
"bloqade": ExperimentType.AHS,
}
if native_alias in native_no_ep:
return native_no_ep[native_alias]
entry_points = get_entrypoints(module)
entry_point = entry_points.get(native_alias)
if entry_point is None:
raise ValueError(f"Entry point '{native_alias}' not found in '{group}'.")
module_path = (
entry_point.value.split(":")[0] if sys.version_info >= (3, 10) else entry_point.module_name
)
programs_sub_module = module_path[len(group) + 1 :].split(".", maxsplit=1)[0]
return ExperimentType(programs_sub_module)