Skip to main contentIBM Quantum Documentation Preview
This is a preview build of IBM Quantum™ documentation. Refer to docs.quantum.ibm.com for the official documentation.

Introduction to Qiskit Functions

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
qiskit-addon-aqc-tensor~=0.1.2
qiskit-addon-obp~=0.1.0
scipy~=1.15.0
pyscf~=2.8.0
Note

Qiskit Functions are an experimental feature available only to IBM Quantum™ Premium Plan users. They are in preview release status and subject to change.

Qiskit Functions simplify and accelerate utility-scale algorithm discovery and application development, by abstracting away parts of the quantum software development workflow. In this way, Qiskit Functions free up time normally spent hand-writing code and fine-tuning experiments.

Overview of Qiskit Functions Functions come in two forms:

TypeWhat does it do?Example inputs and outputsWho is it for?
Circuit functionSimplified interface for running circuits. Abstracts transpilation, error suppression, error mitigationInput: Abstract PUBs
Output: Mitigated expectation values
Researchers using Qiskit to discover new algorithms and applications, without needing to focus on optimizing for hardware or handling error. Circuit functions can be used to build custom application functions.
Application functionCovers higher-level tasks, like exploring algorithms and domain-specific use cases. Abstracts quantum workflow to solve tasks, with classical inputs and outputsInput: Molecules, graphs
Output: Energy, cost
Researchers in non-quantum domains, integrating quantum into existing large-scale classical workflows, without needing to map classical data to quantum circuits.

Functions are provided by IBM® and third-party partners. Each is performant for specific workload characteristics and have unique performance-tuning options. Premium Plan users can get started with IBM Qiskit Functions for free, or procure a license from one of the partners who have contributed a function to the catalog.


Get started with Qiskit Functions

Install Qiskit Functions Catalog client

  1. To start using Qiskit Functions, install the IBM Qiskit Functions Catalog client:

    pip install qiskit-ibm-catalog
    
  2. Retrieve your API token from the IBM Quantum account page, and activate your Python virtual environment. See the installation instructions if you do not already have a virtual environment set up.

    If you are working in a trusted Python environment (such as on a personal laptop or workstation), use the save_account() method to save your credentials locally. (Skip to the next step if you are not using a trusted environment, such as a shared or public computer, to authenticate to IBM Quantum Platform.)

    To use save_account(), run python in your shell to open a REPL (read-eval-print loop), then enter the following:

    from qiskit_ibm_catalog import QiskitFunctionsCatalog
     
    QiskitFunctionsCatalog.save_account(token="<your-token>")

    Close out of the REPL with exit(). From now on, whenever you need to authenticate to the service, you can load your credentials with QiskitFunctionsCatalog().

# Load saved credentials
from qiskit_ibm_catalog import QiskitFunctionsCatalog
 
catalog = QiskitFunctionsCatalog()
  1. Avoid executing code on an untrusted machine or an external cloud Python environment to minimize security risks. If you must use an untrusted environment (on, for example, a public computer), change your API token after each use by expiring it on the IBM Quantum Platform dashboard (click the refresh button in the API token field) to reduce risk. To initialize the service in this situation, expand the following section to view code you can use:

    Initialize the service in an untrusted environment
    from qiskit_ibm_catalog import QiskitFunctionsCatalog
     
    # After using the following code, go to your dashboard (https://quantum.ibm.com/)
    # and expire your API token (click the refresh button in the API token field)
    catalog = QiskitFunctionsCatalog(token="<MY_IBM_QUANTUM_TOKEN>")
    Caution

    Protect your API token! Never include your token in source code, Python script, or notebook file. When sharing code with others, ensure that your API token is not embedded directly within the Python script. Instead, share the script without the token and provide instructions for securely setting it up.

    If you accidentally share your token with someone or include it in version control like Git, immediately revoke your token by expiring it on the IBM Quantum Platform dashboard (click the refresh button in the API token field) to reduce risk.

  2. Once you have authenticated, you can list the functions from the Qiskit Functions Catalog that you have access to:

catalog.list()

Output:

[QiskitFunction(qunasys/quri-chemistry),
 QiskitFunction(algorithmiq/tem),
 QiskitFunction(qedma/qesem),
 QiskitFunction(ibm/circuit-function),
 QiskitFunction(q-ctrl/optimization-solver),
 QiskitFunction(q-ctrl/performance-management)]

Run enabled functions

Once a catalog object has been instantiated, you can select a function using catalog.load(provider/function-name):

ibm_cf = catalog.load("ibm/circuit-function")

Each Qiskit Function has custom inputs, options, and outputs. Check the specific documentation pages for the function you want to run for more information. By default, all users can only run one function job at a time:

job = ibm_cf.run(
    pubs=[(circuit, observable)],
    instance=instance,  # E.g. "ibm-q/open/main"
    backend_name=backend_name,  # E.g. "ibm_kyiv"
)
 
job.job_id

Output:

'b6a4b948-b8d3-40dc-848e-3a973e6ac29f'

Check job status

Tip

Currently, the IBM Quantum workloads table only reflects Qiskit Runtime workloads. Use job.status() to see your Qiskit Function workload's current status.

With your Qiskit Function job_id, you can check the status of running jobs. This includes the following statuses:

  • QUEUED: The remote program is in the Qiskit Function queue. The queue priority is based on how much you've used Qiskit Functions.
  • INITIALIZING: The remote program is starting; this includes setting up the remote environment and installing dependencies.
  • RUNNING: The program is running.
  • DONE: The program is complete, and you can retrieve result data with job.results().
  • ERROR: The program stopped running because of a problem. Use job.result() to get the error message.
  • CANCELED: The program was canceled; either by a user, the service, or the server.
job.status()

Output:

'QUEUED'

Retrieve results

Once a program is DONE, you can use job.results() to fetch the result. This output format varies with each function, so be sure to follow the specific documentation:

result = job.result()
print(result)

Output:

PrimitiveResult([PubResult(data=DataBin(evs=np.ndarray(<shape=(), dtype=float64>), stds=np.ndarray(<shape=(), dtype=float64>), ensemble_standard_error=np.ndarray(<shape=(), dtype=float64>)), metadata={'shots': 4096, 'target_precision': 0.015625, 'circuit_metadata': {}, 'resilience': {}, 'num_randomizations': 32})], metadata={'dynamical_decoupling': {'enable': True, 'sequence_type': 'XX', 'extra_slack_distribution': 'middle', 'scheduling_method': 'alap'}, 'twirling': {'enable_gates': False, 'enable_measure': True, 'num_randomizations': 'auto', 'shots_per_randomization': 'auto', 'interleave_randomizations': True, 'strategy': 'active-accum'}, 'resilience': {'measure_mitigation': True, 'zne_mitigation': False, 'pec_mitigation': False}, 'version': 2})

At any time, you can also cancel a job:

job.stop()

Output:

'Job has been stopped.'

List previously run jobs run with Qiskit Functions

You can use jobs() to list all jobs submitted to Qiskit Functions:

old_jobs = catalog.jobs()
old_jobs

Output:

[<Job | cd38fff2-0531-4398-93af-4a03c8e35666>,
 <Job | 98c372f9-b9e3-4f32-a17f-31c820461c6c>,
 <Job | b6a4b948-b8d3-40dc-848e-3a973e6ac29f>,
 <Job | 9f2c9f34-010e-46e9-833f-e2e49b74f8f1>,
 <Job | 190922c7-dd9f-48ad-9a23-6fc4e5f0963d>,
 <Job | caa33021-d7b5-48a5-96c2-0661001114da>,
 <Job | 7159d742-3887-42c8-9b94-c9f437ee308d>,
 <Job | 6c27d6df-e131-4de8-a042-0abf428e197a>,
 <Job | ee1506e1-6745-4fa4-babd-ab959d4ee7dd>,
 <Job | 8194c6c5-8774-4787-9d5d-78b46260e74b>]

Fetch error messages

If a program status is ERROR, use job.result() to fetch the error message to help debug as follows:

print(job.result())

Output:

PrimitiveResult([PubResult(data=DataBin(evs=np.ndarray(<shape=(), dtype=float64>), stds=np.ndarray(<shape=(), dtype=float64>), ensemble_standard_error=np.ndarray(<shape=(), dtype=float64>)), metadata={'shots': 4096, 'target_precision': 0.015625, 'circuit_metadata': {}, 'resilience': {}, 'num_randomizations': 32})], metadata={'dynamical_decoupling': {'enable': True, 'sequence_type': 'XX', 'extra_slack_distribution': 'middle', 'scheduling_method': 'alap'}, 'twirling': {'enable_gates': False, 'enable_measure': True, 'num_randomizations': 'auto', 'shots_per_randomization': 'auto', 'interleave_randomizations': True, 'strategy': 'active-accum'}, 'resilience': {'measure_mitigation': True, 'zne_mitigation': False, 'pec_mitigation': False}, 'version': 2})

Next steps

Recommendations