Skip to content

Latest commit

 

History

History
1362 lines (959 loc) · 38 KB

README.md

File metadata and controls

1362 lines (959 loc) · 38 KB

Quantum Programming Studio API

Python wrapper for Quantum Programming Studio HTTP API.

Install and configure

1. Install QPS API package:

pip install quantastica-qps-api

2. Find your QPS API token:

Login to Quantum Programming Studio, go to Profile -> API Access and copy your API token.

3. Configure QPS API package with your API token:

from quantastica.qps_api import QPS

QPS.save_account("YOUR_API_TOKEN")

That will create a local configuration file where your API token will be stored for future use.

Now you are ready to use QPS API.

Account management

QPS.save_account(api_token, api_url=None)

Run this once to setup your QPS REST API account. Method will create configuration file and your api token will be stored there for future use.

If needed, you can clear your token by running QPS.save_account("") (or by deleting a configuration file).

If api_url is not provided then https://quantum-circuit.com/api/ will be set as default.

QPS.config_path()

You can get config file path by running QPS.config_path().

Default configuration file path:

  • On Unix, directory is obtained from environment variable HOME if it is set; otherwise the current user’s home directory is looked up in the password directory through the built-in module pwd.

  • On Windows, USERPROFILE will be used if set, otherwise a combination of HOMEPATH and HOMEDRIVE will be used.

Synthesis and transpilation API

Synthesis and transpilation tool can be used to:

  • create quantum circuit from state vectors

  • create quantum circuit from truth table

  • create quantum circuit from unitary matrix (decompose unitary matrix)

  • transpile circuits (change instruction set)

Circuit from vectors

Find quantum circuit from pairs of initial & final state vectors (wave functions).

QPS.synth.circuit_from_vectors(vector_pairs, endianness = "little", job_name=None, settings = {}, start_job=True)

  • vector_pairs is list containing vector pairs. Each vector pair is list with 2 elements: initial vector and final vector. All vectors in all pairs must be of same length (same number of qubits).

  • endianness string. Orientation of bits in state vector (most significant bit/first qubit or least significant bit/first qubit). Can be little (like Qiskit) or big. Default is little.

  • job_name string is optional. You can give it a human readable name.

  • start_job if this argument is True (default) the job will be immediatelly sent to execution queue. If start_job is False then it will stay in draft state and you will be able to start it later by calling start_job() method.

  • settings object is optional. Default is:

{
	"strategy": "strategy_a",
	"pre_processing": "",
	"allowed_gates": "u3,cx",
	"min_gates": 0,
	"max_gates": 0,
	"max_diff": 0.001,
	"diff_method": "ignorephase",
	"max_duration": 0,
	"single_solution": True
}

Settings

  • strategy string. Can be one of: strategy_a (brute force or heuristics) and strategy_b OptigenQ+QSD (OptigenQ is Quantastica's method to find unitary matrix from pairs of vectors, and QSD is used to decompose matrix and return circuit). Default is strategy_a.

  • pre_processing string. Used only with strategy_a (brute force or heuristics). Can be one of: stable, experimental_1, experimental_2, experimental_3, experimental_5. Default is empty string (Empty string means "use default" which is currently stable).

  • allowed_gates string. Used only with strategy_a (brute force or heuristics). Comma delimited gate names (instruction set) to use. Default is u3,cx. With strategy_b instruction set is fixed to u3,cx (you can transpile returned circuit later using QPS.synth.transpile() method.

  • min_gates and max_gates integer. Used only with strategy_a (brute force or heuristics). Default is 0 (no limits).

  • max_diff float. Used only with strategy_a (brute force or heuristics). Default is 0.001 (1e-3).

  • diff_method string. Used only with strategy_a (brute force or heuristics). Can be one of: distance (exact match), ignorephase (match up to a global phase) and abs (match absolute values). Default is ignorephase.

  • max_duration integer. Timeout in seconds. Solver will stop after specified number of seconds and error will be returned. Useful with brute force algorithm which has high computational complexity and can run for a very long time. You can decide when to stop (and for example proceed with the next job which is using different method and is waiting in a queue - if you prepared it). Default is 0 which means "maximum allowed by your subscription plan". Free plan has limit of a few seconds (subject to change).

  • single_solution boolean. Used only with strategy_a (brute force or heuristics). When True, solver will stop when first solution was found. When False solver will return all possible configurations of a circuit.

Note: if settings argument is provided, it will overwrite default settings, but only provided keys will be overwritten - not entire default settings object.

Example:

from quantastica.qps_api import QPS

vector_pairs = [
[ [1, 0, 0, 0], [ 0.5+0j,  0.5+0j,  0.5+0j,  0.5+0j ] ],
[ [0, 1, 0, 0], [ 0.5+0j,  0+0.5j, -0.5+0j,  0-0.5j ] ],
[ [0, 0, 1, 0], [ 0.5+0j, -0.5+0j,  0.5+0j, -0.5+0j ] ],
[ [0, 0, 0, 1], [ 0.5+0j,  0-0.5j, -0.5+0j,  0+0.5j ] ]
]

job_id = QPS.synth.circuit_from_vectors(vector_pairs, settings = { "instruction_set": ["h", "cu1", "swap"], "single_solution": False })

job = QPS.synth.get_job(job_id, wait=True)

job_status = job["status"]
job_output = job["output"]

if(job_status == "error"):
	print(job_output["message"])
else:
	for circuit in job_output["circuits"]:
		print(circuit["qasm"])

Example output:

OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
h q[1];
swap q[0], q[1];
cu1 (2.356194496154785) q[0], q[1];
h q[1];

OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
h q[1];
cu1 (2.356194496154785) q[0], q[1];
h q[0];
swap q[0], q[1];

OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
h q[1];
cu1 (2.356194496154785) q[0], q[1];
swap q[0], q[1];
h q[1];

OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
swap q[0], q[1];
h q[0];
cu1 (2.356194496154785) q[0], q[1];
h q[1];

State preparation

Get circuit which prepares provided state.

QPS.synth.state_preparation(final_vector, endianness = "little", job_name=None, settings = {}, start_job=True)

  • final_vector is target vector.

  • endianness string. Orientation of bits in state vector (most significant bit/first qubit or least significant bit/first qubit). Can be little (like Qiskit) or big. Default is little.

  • job_name string is optional. You can give it a human readable name.

  • start_job if this argument is True (default) the job will be immediatelly sent to execution queue. If start_job is False then it will stay in draft state and you will be able to start it later by calling start_job() method.

  • settings object is optional. Default is:

{
	"strategy": "strategy_a",
	"pre_processing": "",
	"allowed_gates": "u3,cx",
	"min_gates": 0,
	"max_gates": 0,
	"max_diff": 0.001,
	"diff_method": "ignorephase",
	"max_duration": 0,
	"single_solution": True
}

Settings

  • strategy string. Can be one of: strategy_a (brute force or heuristics) and strategy_b OptigenQ+QSD (OptigenQ is Quantastica's method to find unitary matrix from pairs of vectors, after which QSD is used to decompose matrix and return circuit). Default is strategy_a.

  • pre_processing string. Used only with strategy_a (brute force or heuristics). Can be one of: stable, experimental_1, experimental_2, experimental_3, experimental_5. Default is empty string (Empty string means "use default" which is currently stable).

  • allowed_gates string. Used only with strategy_a (brute force or heuristics). Comma delimited gate names (instruction set) to use. Default is u3,cx. With strategy_b instruction set is fixed to u3,cx (you can transpile returned circuit later using QPS.synth.transpile() method.

  • min_gates and max_gates integer. Used only with strategy_a (brute force or heuristics). Default is 0 (no limits).

  • max_diff float. Used only with strategy_a (brute force or heuristics). Default is 0.001 (1e-3).

  • diff_method string. Used only with strategy_a (brute force or heuristics). Can be one of: distance (exact match), ignorephase (match up to a global phase) and abs (match absolute values). Default is ignorephase.

  • max_duration integer. Timeout in seconds. Solver will stop after specified number of seconds and error will be returned. Useful with brute force algorithm which has high computational complexity and can run for a very long time. You can decide when to stop (and for example proceed with the next job which is using different method and is waiting in a queue - if you prepared it). Default is 0 which means "maximum allowed by your subscription plan". Free plan has limit of a few seconds (subject to change).

  • single_solution boolean. Used only with strategy_a (brute force or heuristics). When True, solver will stop when first solution was found. When False solver will return all possible configurations of a circuit.

Note: if settings argument is provided, it will overwrite default settings, but only provided keys will be overwritten - not entire default settings object.

Example:

from quantastica.qps_api import QPS

desired_state = [0.5, 0.5, 0.5, 0.5]

job_id = QPS.synth.state_preparation(desired_state, settings = { "instruction_set": ["u3", "cx"] })

job = QPS.synth.get_job(job_id, wait=True)

job_status = job["status"]
job_output = job["output"]

if(job_status == "error"):
	print(job_output["message"])
else:
	for circuit in job_output["circuits"]:
		print(circuit["qasm"])

Example output:

OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
u3 (1.570796370506287, 0, 1.217840194702148) q[0];
u3 (1.570796370506287, 0, 0.621559917926788) q[1];

Transpile

Transpile circuit (change instruction set).

QPS.synth.transpile(input_qasm, method="replace_blocks", method_options={}, job_name=None, settings = {}, start_job=True)

  • input_qasm is string containing OpenQASM 2.0 code.

  • method is method name string. Can be one of: "replace_circuit", "replace_blocks", "replace_gates". Default: "replace_blocks".

  • method_options dict with following structure:

    • If method is replace_blocks then: { "block_size": 2, "two_pass": False } (maximum block size is 4).

    • For other methods: no options (method_options is ignored)

  • job_name string is optional. You can give it a human readable name.

  • start_job if this argument is True (default) the job will be immediatelly sent to execution queue. If start_job is False then it will stay in draft state and you will be able to start it later by calling start_job() method.

  • settings object is optional. Default is:

{
	"pre_processing": "experimental1",
	"allowed_gates": "u3,cx",
	"max_diff": 0.001,
	"diff_method": "ignorephase",
	"max_duration": 0
}

Settings

  • pre_processing string. Can be one of: stable, experimental_1, experimental_2, experimental_3, experimental_5. Default is empty string (Empty string means "use default" which is currently stable).

  • allowed_gates string. Comma delimited gate names (instruction set) to use. Default is u3,cx.

  • max_diff float. Default is 0.001 (1e-3).

  • diff_method string. Can be one of: distance (exact match), ignorephase (match up to a global phase) and abs (match absolute values). Default is ignorephase.

  • max_duration integer. Timeout in seconds. Solver will stop after specified number of seconds and error will be returned. Default is 0 which means "maximum allowed by your subscription plan". Free plan has limit of a few seconds (subject to change).

Note: if settings argument is provided, it will overwrite default settings, but only provided keys will be overwritten - not entire default settings object.

Example:

from quantastica.qps_api import QPS

input_qasm = """
OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
h q[0];
cx q[0], q[1];
"""

job_id = QPS.synth.transpile(input_qasm, settings = { "instruction_set": ["id", "x", "sx", "rz", "cx"], "diff_method": "ignorephase" })

job = QPS.synth.get_job(job_id, wait=True)

job_status = job["status"]
job_output = job["output"]

if(job_status == "error"):
	print(job_output["message"])
else:
	for circuit in job_output["circuits"]:
		print(circuit["qasm"])

Example output:

OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
sx q[0];
rz (1.570796370506287) q[0];
sx q[0];
cx q[0], q[1];

Decompose matrix

Decompose unitary matrix (find circuit from matrix).

QPS.synth.decompose_unitary(unitary, endianness="big", job_name=None, settings = {}, start_job=True)

  • unitary matrix operator.

  • endianness - orientation of the matrix. Can be little endian (like Qiskit) or big endian. Default is big. Note that default endianness of the matrix differs from default endianness of vectors in other methods. That's to be aligned with QPS. In Qiskit, both matrices and vectors are little endian. So, if you are solving unitary from Qiskit then provide endianness = "little" argument.

  • job_name string is optional. You can give it a human readable name.

  • start_job if this argument is True (default) the job will be immediatelly sent to execution queue. If start_job is False then it will stay in draft state and you will be able to start it later by calling start_job() method.

  • settings object is optional. Default is:

{
	"strategy": "strategy_a",
	"pre_processing": "",
	"allowed_gates": "u3,cx",
	"coupling_map": [],
	"min_gates": 0,
	"max_gates": 0,
	"max_diff": 0.001,
	"diff_method": "ignorephase",
	"max_duration": 0,
	"single_solution": True
}

Settings

  • strategy string. Can be one of: strategy_a (brute force or heuristics) and strategy_b QSD. Default is strategy_a. Note that strategy_a (brute force or heuristics) returns circuit with optimal or near optimal depth, but this method has very high computational complexity and can run for a very long time (depending on number of qubits and total number of gates in the resulting circuit).

  • pre_processing string. Used only with strategy_a (brute force or heuristics). Can be one of: stable, experimental_1 and experimental_5. Default is empty string (Empty string means "use default" which is currently stable). Experimental methods have lower computational complexity and should finish sooner, but that depends on many factors, so you should try all methods and see what works best for your particular problem.

  • allowed_gates string. Used only with strategy_a (brute force or heuristics). Comma delimited gate names (instruction set) to use. Default is u3,cx. With strategy_b instruction set is fixed to u3,cx (you can transpile returned circuit later using QPS.synth.transpile() method.

  • min_gates and max_gates integer. Used only with strategy_a (brute force or heuristics). Default is 0 (no limits).

  • max_diff float. Used only with strategy_a (brute force or heuristics). Default is 0.001 (1e-3).

  • diff_method string. Used only with strategy_a (brute force or heuristics). Can be one of: distance (exact match), ignorephase (match up to a global phase) hs (match up to a global phase using faster method) and abs (match absolute values). Default is ignorephase.

  • max_duration integer. Timeout in seconds. Solver will stop after specified number of seconds and error will be returned. Useful with brute force algorithm which has high computational complexity and can run for a very long time. You can decide when to stop (and for example proceed with the next job which is using different method and is waiting in a queue - if you prepared it). Default is 0 which means "maximum allowed by your subscription plan". Free plan has limit of a few seconds (subject to change).

  • single_solution boolean. Used only with strategy_a (brute force or heuristics). When True, solver will stop when first solution was found. When False solver will return all possible configurations of a circuit.

Note: if settings argument is provided, it will overwrite default settings, but only provided keys will be overwritten - not entire default settings object.

Example:

from quantastica.qps_api import QPS

unitary = [
[ 0.5+0.0j,  0.5+0.0j,  0.5+0.0j,  0.5+0.0j],
[ 0.5+0.0j,  0.5+0.0j, -0.5+0.0j, -0.5+0.0j],
[ 0.5+0.0j, -0.5+0.0j,  0.0+0.5j,  0.0-0.5j],
[ 0.5+0.0j, -0.5+0.0j,  0.0-0.5j,  0.0+0.5j]
]

job_id = QPS.synth.decompose_unitary(unitary, settings = { "instruction_set": ["h", "cu1", "swap"], "single_solution": False })

job = QPS.synth.get_job(job_id, wait=True)

job_status = job["status"]
job_output = job["output"]

if(job_status == "error"):
	print(job_output["message"])
else:
	for circuit in job_output["circuits"]:
		print(circuit["qasm"])

Example output:

OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
h q[1];
swap q[0], q[1];
cu1 (1.570796370506287) q[0], q[1];
h q[1];

OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
h q[1];
cu1 (1.570796370506287) q[0], q[1];
h q[0];
swap q[0], q[1];

OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
h q[1];
cu1 (1.570796370506287) q[0], q[1];
swap q[0], q[1];
h q[1];

OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
swap q[0], q[1];
h q[0];
cu1 (1.570796370506287) q[0], q[1];
h q[1];

Create algorithm from truth table

Create circuit which implements logical expression whose truth table is given.

QPS.synth.circuit_from_truth_table(truth_table_csv, column_defs, csv_delimiter=None, additional_qubits=1, job_name=None, settings={}, start_job=True)

  • truth_table_csv is string containing truth table in CSV format

  • column_defs list of strings describing each column from truth table: "input", "output" or "ignore"

  • csv_delimiter CSV column delimiter char: None, "," (comma) or "\t" (tab). If delimiter is None (default) it will be automatically detected.

  • additional_qubits number of qubits to add (to displace input and output qubits).

  • job_name string is optional. You can give it a human readable name.

  • settings object is optional. Default is:

{
	"pre_processing": "",
	"allowed_gates": "x,cx,ccx,swap",
	"coupling_map": [],
	"min_gates": 0,
	"max_gates": 0,
	"max_diff": 0.001,
	"diff_method": "ignorephase",
	"max_duration": 0,
	"single_solution": True
}

Settings

  • pre_processing string. Can be one of: stable, experimental_1 and experimental_5. Default is empty string (Empty string means "use default" which is currently stable).

  • allowed_gates string. Comma delimited gate names (instruction set) to use. Default is x,cx,ccx,swap.

  • min_gates and max_gates integer. Default is 0 (no limits).

  • max_diff float. Used only with strategy_a (brute force or heuristics). Default is 0.001 (1e-3).

  • diff_method string. Used only with strategy_a (brute force or heuristics). Can be one of: distance (exact match), ignorephase (match up to a global phase) and abs (match absolute values). Default is ignorephase.

  • max_duration integer. Timeout in seconds. Solver will stop after specified number of seconds and error will be returned. Useful with brute force algorithm which has high computational complexity and can run for a very long time. You can decide when to stop (and for example proceed with the next job which is using different method and is waiting in a queue - if you prepared it). Default is 0 which means "maximum allowed by your subscription plan". Free plan has limit of a few seconds (subject to change).

  • single_solution boolean. Used only with strategy_a (brute force or heuristics). When True, solver will stop when first solution was found. When False solver will return all possible configurations of a circuit.

Note: if settings argument is provided, it will overwrite default settings, but only provided keys will be overwritten - not entire default settings object.

Example:

from quantastica.qps_api import QPS

truth_table = """
A,B,A_NAND_B
0,0,1
0,1,1
1,0,1
1,1,0
"""

job_id = QPS.synth.circuit_from_truth_table(truth_table, ["input", "input", "output"])

job = QPS.synth.get_job(job_id, wait=True)

job_status = job["status"]
job_output = job["output"]

if(job_status == "error"):
	raise Exception(job_output["message"])
else:
	if(len(job_output["circuits"]) == 0):
		raise Exception("No results.")
	else:
		for circuit in job_output["circuits"]:
			print(circuit["qasm"])

Example output:

OPENQASM 2.0;
include "qelib1.inc";
qreg q[3];
x q[2];
ccx q[0], q[1], q[2];

Run problem file

Solve problem provided in internal format used by synthesizer and transpiler.

QPS.synth.solve(problem, settings = {}, start_job=True)

  • problem object (e.g. job exported to json from Quantum Programming studio).

  • settings argument is optional. If provided, it will overwrite keys in problem.settings. Note that only provided keys will be overwritten - not entire problem.settings object.

  • start_job if this argument is True (default) the job will be immediatelly sent to execution queue. If start_job is False then it will stay in draft state and you will be able to start it later by calling start_job() method.

Example:

from quantastica.qps_api import QPS

problem = {
	"name": "Bell",
	"type": "vectors",
	"source": {
		"vectors": {
			"text1": "[ 1, 0, 0, 0 ]",
			"text2": "[ 1/sqrt(2), 0, 0, 1/sqrt(2) ]",
			"endianness1": "little",
			"endianness2": "little"
		}
	},
	"problem": [
		{
			"input": [
				1,
				0,
				0,
				0
			],
			"output": [
				0.7071067811865475,
				0,
				0,
				0.7071067811865475
			]
		}
	],
	"settings": {
		"strategy": "strategy_a",
		"pre_processing": "",
		"allowed_gates": "u3,cx",
		"coupling_map": [],
		"min_gates": 0,
		"max_gates": 0,
		"max_diff": 0.001,
		"diff_method": "distance",
		"max_duration": 0,
		"single_solution": False
	}
}

job_id = QPS.synth.solve(problem)

job = QPS.synth.get_job(job_id, wait=True)

job_status = job["status"]
job_output = job["output"]

if(job_status == "error"):
	print(job_output["message"])
else:
	for circuit in job_output["circuits"]:
		print(circuit["qasm"])

Example output:

OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
u3 (-1.570796370506287, -3.141592741012573, -2.675650835037231) q[0];
cx q[0], q[1];

OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
u3 (-1.570796370506287, -3.141592741012573, -2.675650835037231) q[1];
cx q[1], q[0];

Synthesizer/transpiler output format

Finished job object has following structure:

{
	"name"       : String,
	"type"       : String,
	"source"     : Object,
	"problem"    : Array,
	"settings"   : Object,
	"status"     : String,
	"output"     : Object,
	"queuedAt"   : String,
	"startedAt"  : String,
	"finishedAt" : String
}

Keys important to user are:

  • name String: name of the job

  • status String: can be draft, queued, running, error, done.

  • output Object with following structure:

{
	"error_code" : Integer,
	"message"    : String,
	"time_taken" : Float,
	"version"    : String,
	"circuits"   : Array of Object
}
  • error_code Integer: 0 on success, non-zero on error

  • message String: error message if error code is non-zero

  • time_taken Float: number of seconds

  • version String: solver version

  • circuits Array: resulting circuits. Each is object with following structure:

{
	"qubits"  : Integer,
	"cregs"   : Array,
	"program" : Array,
	"diff"    : Float,
	"index"   : Integer,
	"qasm"    : String,
	"qasmExt" : String
}

Keys important to user are:

  • qasm OpenQASM 2.0 source code of the resulting circuit.

  • qasmExt OpenQASM 2.0 with extended instruction set (all gates supported by Quantum Programming Studio).

Difference between qasm and qasmExt: if circuit contains gate supported by QPS but not directly supported by OpenQASM 2.0 then qasm will contain equivalent circuit transpiled to OpenQASM 2.0 instruction set, but qasmExt will contain gates as is.

For example if circuit contains IONQ native gate gpi2(2.51678906856393) on first qubit:

qasm will be:

OPENQASM 2.0;
include "qelib1.inc";
qreg q[1];
u3 (1.5707963267948966, 0.9459927417690333, -0.9459927417690333) q[0];

qasmExt will contain:

OPENQASM 2.0;
include "qelib1.inc";
qreg q[1];
gpi2 (2.51678906856393) q[0];

Example job object with output:

{
	"_id": "r9LskFoLPQW5w7HTp",
	"name": "Bell",
	"type": "vectors",
	"source": {
		"vectors": {
			"text1": "[ 1, 0, 0, 0 ]",
			"text2": "[ 1/sqrt(2), 0, 0, 1/sqrt(2) ]",
			"endianness1": "little",
			"endianness2": "little"
		}
	},
	"problem": [
		{
			"input": [
				1,
				0,
				0,
				0
			],
			"output": [
				0.7071067811865475,
				0,
				0,
				0.7071067811865475
			]
		}
	],
	"settings": {
		"strategy": "strategy_a",
		"pre_processing": "",
		"allowed_gates": "u3,cx",
		"coupling_map": [],
		"min_gates": 0,
		"max_gates": 0,
		"max_diff": 0.001,
		"diff_method": "distance",
		"max_duration": 0,
		"single_solution": False
	},
	"output": {
		"circuits": [
			{
				"qubits": 2,
				"cregs": [],
				"diff": 0,
				"program": [
					{
						"name": "u3",
						"wires": [
							0
						],
						"options": {
							"params": {
								"theta": 1.570796326794896,
								"phi": 0,
								"lambda": 1.456034103897321
							}
						}
					},
					{
						"name": "cx",
						"wires": [
							0,
							1
						],
						"options": {}
					}
				],
				"index": 0
			},
			{
				"qubits": 2,
				"cregs": [],
				"diff": 0,
				"program": [
					{
						"name": "u3",
						"wires": [
							1
						],
						"options": {
							"params": {
								"theta": 1.570796326794896,
								"phi": 0,
								"lambda": 1.456034103897321
							}
						}
					},
					{
						"name": "cx",
						"wires": [
							1,
							0
						],
						"options": {}
					}
				],
				"index": 1
			}
		],
		"error_code": 0,
		"message": "",
		"time_taken": 0.008,
		"version": "0.1.0",
		"stats": {
			"result_count": 2
		}
	},
	"createdAt": "2021-02-06T23:39:29.108Z",
	"modifiedAt": "2021-02-06T23:39:30.383Z",
	"queuedAt": "2021-02-06T23:39:29.676Z",
	"startedAt": "2021-02-06T23:39:29.926Z",
	"finishedAt": "2021-02-06T23:39:30.383Z",
	"status": "done"
}

Using synthesizer and transpiler with Qiskit

Format used for input and output is OpenQASM 2.0, so integration with Qiskit (and other frameworks that support OpenQASM) is easy.

Example transpile Qiskit circuit:

from qiskit import QuantumCircuit
from qiskit.circuit.random import random_circuit
from qiskit.quantum_info import Operator

from quantastica.qps_api import QPS

# Generate random Qiskit circuit
qc = random_circuit(5, 5, measure=False)

# Get QASM code
input_qasm = qc.qasm()

# Transpile with QPS
job_id = QPS.synth.transpile(input_qasm, settings = { "instruction_set": ["id", "u3", "cx"], "diff_method": "ignorephase" })
job = QPS.synth.get_job(job_id, wait=True)
job_status = job["status"]
job_output = job["output"]
if(job_status == "error"):
    raise Exception(job_output["message"])

transpiled_circuit = job_output["circuits"][0]

# Get QASM code
transpiled_qasm = transpiled_circuit["qasm"]

# Create Qiskit circuit
transpiled_qc = QuantumCircuit.from_qasm_str(transpiled_qasm)

# Show circuit
print("Depth:", transpiled_qc.depth())
print("Ops:", sum(j for i, j in transpiled_qc.count_ops().items()))
display(transpiled_qc.draw(output="mpl"))

Job management

Problem sent to solver (synthesizer or transpiler) is called a "job". Each job has unique ID. Solver is resource intensive tool, so it is configured to execute only one job at a time. While solver is processing a job, other jobs are queued. When solver finishes a job, it takes the next one from the queue.

API provides functions for job manipulation: you can list all jobs (filtered by status), stop running job, cancel queued jobs, stop/cancel all jobs, start previously canceled (draft) job, etc.

QPS.synth.list_jobs(status_filter=None)

List all jobs, optionally filtered by status.

  • status_filter String, optional. Can be: draft, queued, running, error, done.

Example 1 - list all (unfiltered) jobs:

from quantastica.qps_api import QPS

jobs = QPS.synth.list_jobs()

print(jobs)

Example output:

{
	"list": [
		{ "_id": "r9LskFoLPQW5w7HTp", "name": "Bell state", "type": "vectors", "status": "done" },
		{ "_id": "R8tJH7XoZ233oTREy", "name": "4Q Gauss", "type": "vectors", "status": "queued" },
		{ "_id": "h7fzYbFz8MJvkNhiX", "name": "Challenge", "type": "unitary", "status": "draft" },
		{ "_id": "PC5PNXiGqhh2HmkX8", "name": "Experiment", "type": "vectors", "status": "error"},
		{ "_id": "SNhiCqSCT2WwRWKCd", "name": "Decompose", "type": "unitary", "status": "running" }
	]
}

Example 2 - list running jobs:

from quantastica.qps_api import QPS

jobs = QPS.synth.list_jobs(status_filter="running")

print(jobs)

Example output:

{
	"list": [
		{ "_id": "SNhiCqSCT2WwRWKCd", "name": "Decompose", "type": "unitary", "status": "running" }
	]
}

QPS.synth.job_status(job_id)

Get job status.

Example:

from quantastica.qps_api import QPS

status = QPS.synth.job_status("PC5PNXiGqhh2HmkX8")

print(status)

Example output:

{ "_id": "PC5PNXiGqhh2HmkX8", "name": "Experiment", "type": "vectors", "status": "error", "message": "connect ECONNREFUSED" }

QPS.synth.get_job(job_id, wait=True)

Get job referenced by ID. If wait argument is True (default), then function will wait for a job to finish (or fail) before returning. If wait is False, then job will be immediatelly returned even if it is still running (in which case it will not contain a solution).

Example:

from quantastica.qps_api import QPS

job = QPS.synth.get_job("r9LskFoLPQW5w7HTp")

print(job)

Example output:

{
	"_id": "r9LskFoLPQW5w7HTp",
	"name": "Bell",
	"type": "vectors",
	"source": {
		"vectors": {
			"text1": "[ 1, 0, 0, 0 ]",
			"text2": "[ 1/sqrt(2), 0, 0, 1/sqrt(2) ]",
			"endianness1": "little",
			"endianness2": "little"
		}
	},
	"problem": [
		{
			"input": [ 1, 0, 0, 0 ],
			"output": [ 0.7071067811865475, 0, 0, 0.7071067811865475 ]
		}
	],
	"settings": {
		"max_duration": 0,
		"allowed_gates": "u3,cx",
		"coupling_map": [],
		"min_gates": 0,
		"max_gates": 0,
		"pre_processing": "",
		"strategy": "strategy_a",
		"max_diff": 0.001,
		"diff_method": "distance",
		"single_solution": False
	},
	"status": "done",
	"output": {
		"circuits": [
			{
				"qubits": 2,
				"cregs": [],
				"diff": 0,
				"program": [
					{
						"name": "u3",
						"wires": [ 0 ],
						"options": {
							"params": {
								"theta": -1.570796370506287,
								"phi": -3.141592741012573,
								"lambda": -5.327113628387451
							}
						}
					},
					{
						"name": "cx",
						"wires": [ 0, 1 ],
						"options": {}
					}
				],
				"index": 0,
				"qasm": "OPENQASM 2.0;\ninclude \"qelib1.inc\";\nqreg q[2];\nu3 (-1.570796370506287, -3.141592741012573, -5.327113628387451) q[0];\ncx q[0], q[1];\n",
				"qasmExt": "OPENQASM 2.0;\ninclude \"qelib1.inc\";\nqreg q[2];\nu3 (-1.570796370506287, -3.141592741012573, -5.327113628387451) q[0];\ncx q[0], q[1];\n"
			},
			{
				"qubits": 2,
				"cregs": [],
				"diff": 0,
				"program": [
					{
						"name": "u3",
						"wires": [ 1 ],
						"options": {
							"params": {
								"theta": -1.570796370506287,
								"phi": -3.141592741012573,
								"lambda": -5.327113628387451
							}
						}
					},
					{
						"name": "cx",
						"wires": [ 1, 0 ],
						"options": {}
					}
				],
				"index": 1,
				"qasm": "OPENQASM 2.0;\ninclude \"qelib1.inc\";\nqreg q[2];\nu3 (-1.570796370506287, -3.141592741012573, -5.327113628387451) q[1];\ncx q[1], q[0];\n",
				"qasmExt": "OPENQASM 2.0;\ninclude \"qelib1.inc\";\nqreg q[2];\nu3 (-1.570796370506287, -3.141592741012573, -5.327113628387451) q[1];\ncx q[1], q[0];\n"				
			}
		],
		"error_code": 0,
		"message": "",
		"time_taken": 0.002,
		"version": "0.1.0"
	},
	"queuedAt": "2021-02-06T23:39:29.676Z",
	"startedAt": "2021-02-06T23:39:29.926Z",
	"finishedAt": "2021-02-06T23:39:30.383Z"
}

QPS.synth.stop_job(job_id)

Stop running or cancel queued job. Job will be put into draft state, and you can start it again later by calling start_job().

Example:

from quantastica.qps_api import QPS

response = QPS.synth.stop_job("SNhiCqSCT2WwRWKCd")

print(response)

Example output:

{ "_id": "SNhiCqSCT2WwRWKCd", message: "OK" }

QPS.synth.stop_all_jobs(status_filter=None)

Stop running job / cancel all queued jobs.

  • status_filter - you can stop only a running job by providing status_filter="running" (after this, next job from the queue will be executed). Or, you can cancel all queued jobs by providing status_filter="queued" (running job will not be affected - it will continue running).

Example 1 - stop running job and remove all jobs from queue:

from quantastica.qps_api import QPS

stopped = QPS.synth.stop_all_jobs()

print(stopped)

Example output:

{
	"stopped": [ 
		{ "_id": "SNhiCqSCT2WwRWKCd", "name": "Decompose", "type": "unitary" },
		{ "_id": "R8tJH7XoZ233oTREy", "name": "4Q Gauss", "type": "vectors" }
	]
}

Example 2 - stop only a running job. Next job from queue, if any, will start:

from quantastica.qps_api import QPS

stopped = QPS.synth.stop_all_jobs(status_filter="running")

print(stopped)

Example output:

{
	"stopped": [
		{ "_id": "SNhiCqSCT2WwRWKCd", "name": "Decompose", "type": "unitary" }
	]
}

Example 3 - cancel all queued jobs. Running job will not be affected:

from quantastica.qps_api import QPS

stopped = QPS.synth.stop_all_jobs(status_filter="queued")

print(stopped)

Example output:

{
	"stopped": [
		{ "_id": "R8tJH7XoZ233oTREy", "name": "4Q Gauss", "type": "vectors" }
	]
}

QPS.synth.start_job(job_id)

Start previously stopped/canceled job (can be any job with status draft).

Example:

from quantastica.qps_api import QPS

response = QPS.synth.start_job("SNhiCqSCT2WwRWKCd")

print(response)

Example output:

{ "_id": "SNhiCqSCT2WwRWKCd", "message": "OK" }

Quantum Language Converter API

Quantum Language Converter is a tool which converts quantum program between different quantum programming languages and frameworks. It is also available as a q-convert command line tool and as a web UI at https://quantum-circuit.com/qconvert.

QPS has integrated quantum language converter API which you can access directly from python code:

QPS.converter.convert(input, source, dest)

Converts input quantum program given as string from source format into dest format.

Example 1 - convert QASM 2.0 program to QUIL:

from quantastica.qps_api import QPS

input_program = """
OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
creg c[2];
h q[0];
cx q[0], q[1];
measure q[0] -> c[0];
measure q[1] -> c[1];
"""

output_program = QPS.converter.convert(input_program, "qasm", "quil")

print(output_program)

Output:

DECLARE ro BIT[2]
H 0
CNOT 0 1
MEASURE 0 ro[0]
MEASURE 1 ro[1]

Example 2 - convert QASM 2.0 program to circuit drawing as vector image:

from quantastica.qps_api import QPS

input_program = """
OPENQASM 2.0;
include "qelib1.inc";
qreg q[2];
creg c[2];
h q[0];
cx q[0], q[1];
measure q[0] -> c[0];
measure q[1] -> c[1];
"""

output_svg = QPS.converter.convert(input_program, "qasm", "svg")

# Do something with returned vector image...
open("output.svg", "w").write(output_svg)

Utils API

QPS.utils.random_circuit(num_qubits=5, output_format="quantum-circuit", options=None)

Returns random quantum circuit.

  • num_qubits Integer. Number of qubits. Default: 5.

  • format String. Output format. The same as QPS.converter.convert() function's dest argument. Example: "qasm". Default: "quantum-circuit".

  • options Dict. Optional. Can contain following keys:

    • instruction_set List of gates to use. Example: ["u3", "cx"]. Default: [ "u3", "rx", "ry", "rz", "cx", "cz" ].
    • num_gates Integer. Number of gates in the circuit. Default is num_qubits * 8.
    • mid_circuit_measurement Bool. Default: False.
    • mid_circuit_reset Bool. Default: False.
    • classic_control Bool. Default: False.