Skip to content

Commit

Permalink
change: Use BraketSimulator.jl in examples, strip out unnecessary `…
Browse files Browse the repository at this point in the history
…PyBraket.jl` components (#84)
  • Loading branch information
kshyatt-aws authored Jun 20, 2024
1 parent 1d05da0 commit f357046
Show file tree
Hide file tree
Showing 27 changed files with 133 additions and 1,388 deletions.
1 change: 1 addition & 0 deletions PyBraket/CondaPkg.toml
Original file line number Diff line number Diff line change
Expand Up @@ -6,3 +6,4 @@ numpy = ""

[pip.deps]
amazon-braket-sdk = ">=1.70.0"
amazon-braket-default-simulator = ">=1.10.0"
7 changes: 3 additions & 4 deletions PyBraket/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,21 +2,20 @@

**PyBraket.jl is not an officially supported AWS product.**

This package provides Julia-Python interoperability between `Braket.jl` and Python features of the Amazon Braket SDK, such as the Amazon Braket [Local Simulators](https://docs.aws.amazon.com/braket/latest/developerguide/braket-send-to-local-simulator.html) and
[Local Jobs](https://docs.aws.amazon.com/braket/latest/developerguide/braket-jobs-local-mode.html).
This package provides Julia-Python interoperability between `Braket.jl` and Python features of the Amazon Braket SDK, such as the Amazon Braket [Local Simulators](https://docs.aws.amazon.com/braket/latest/developerguide/braket-send-to-local-simulator.html) and support for unpickling the [results of Braket Jobs](https://docs.aws.amazon.com/braket/latest/developerguide/braket-jobs-save-results.html).

This is *experimental* software, and support may be discontinued in the future. For a fully supported SDK, please use
the [Python SDK](https://github.com/aws/amazon-braket-sdk-python). We may change, remove, or deprecate parts of the API when making new releases.
Please review the [CHANGELOG](CHANGELOG.md) for information about changes in each release.

## Installation & Prerequisites

You'll need [`PythonCall.jl`](https://cjdoris.github.io/PythonCall.jl).
You'll need [`PythonCall.jl`](https://cjdoris.github.io/PythonCall.jl) installed to use this package.

### Use `CondaPkg.jl`

`PyBraket.jl` now supports [`CondaPkg.jl`](https://github.com/cjdoris/CondaPkg.jl)! The package and its tests come with `CondaPkg.toml` files ready to go.
`CondaPkg` will install all necessary dependencies for you.
`CondaPkg.jl` will install all necessary dependencies for you.

### Use native Python installation

Expand Down
31 changes: 4 additions & 27 deletions PyBraket/src/PyBraket.jl
Original file line number Diff line number Diff line change
Expand Up @@ -5,35 +5,28 @@ module PyBraket
import Braket: Instruction
using Braket.IR
import Braket.IR: TimeSeries, AtomArrangement, DrivingField, PhysicalField, ShiftingField, Setup, Hamiltonian, AHSProgram
export LocalSimulator, PyCircuit
export PyLocalSimulator
import PythonCall: Py

const awsbraket = PythonCall.pynew()
const braketobs = PythonCall.pynew()
const local_sim = PythonCall.pynew()
const default_sim = PythonCall.pynew()
const tasks = PythonCall.pynew()
const circuit = PythonCall.pynew()
const pyjaqcd = PythonCall.pynew()
const pyahs = PythonCall.pynew()
const pyopenqasm = PythonCall.pynew()
const pygates = PythonCall.pynew()
const pynoises = PythonCall.pynew()
const collections = PythonCall.pynew()

function __init__()
# must set these when this code is actually loaded
PythonCall.pycopy!(awsbraket, pyimport("braket.aws"))
PythonCall.pycopy!(braketobs, pyimport("braket.circuits.observables"))
PythonCall.pycopy!(local_sim, pyimport("braket.devices.local_simulator"))
PythonCall.pycopy!(circuit, pyimport("braket.circuits"))
PythonCall.pycopy!(pygates, pyimport("braket.circuits.gates"))
PythonCall.pycopy!(pynoises, pyimport("braket.circuits.noises"))
PythonCall.pycopy!(pyjaqcd, pyimport("braket.ir.jaqcd"))
PythonCall.pycopy!(default_sim, pyimport("braket.default_simulator"))
PythonCall.pycopy!(pyopenqasm, pyimport("braket.ir.openqasm"))
PythonCall.pycopy!(pyahs, pyimport("braket.ir.ahs"))
PythonCall.pycopy!(tasks, pyimport("braket.tasks"))
PythonCall.pycopy!(collections, pyimport("collections"))
PythonCall.pyconvert_add_rule("collections:Counter", Accumulator, counter_to_acc)
Braket._simulator_devices[]["braket_ahs"] = PyLocalSimulator("braket_ahs")
end
function counter_to_acc(::Type{Accumulator}, x::Py)
return counter((pyconvert(Any, el) for el in x.elements()))
Expand Down Expand Up @@ -64,24 +57,8 @@ module PyBraket
end
return args
end

for (irT, pyT) in ((:(Braket.IR.Expectation), :(pyjaqcd.Expectation)),
(:(Braket.IR.Variance), :(pyjaqcd.Variance)),
(:(Braket.IR.Sample), :(pyjaqcd.Sample)),
(:(Braket.IR.Amplitude), :(pyjaqcd.Amplitude)),
(:(Braket.IR.StateVector), :(pyjaqcd.StateVector)),
(:(Braket.IR.Probability), :(pyjaqcd.Probability)),
(:(Braket.IR.DensityMatrix), :(pyjaqcd.DensityMatrix)))
@eval begin
Py(o::$irT) = $pyT(;arg_gen(o, fieldnames($irT))...)
end
end

include("pyahs.jl")
include("pygates.jl")
include("pynoises.jl")
include("pyschema.jl")
using .PySchema
include("pycircuit.jl")
include("local_simulator.jl")
end
78 changes: 34 additions & 44 deletions PyBraket/src/local_simulator.jl
Original file line number Diff line number Diff line change
@@ -1,53 +1,43 @@
struct LocalQuantumTask
id::String
result::Braket.AbstractQuantumTaskResult
end
Braket.state(b::LocalQuantumTask) = "COMPLETED"
Braket.id(b::LocalQuantumTask) = b.id
Braket.result(b::LocalQuantumTask) = b.result


struct LocalSimulator <: Braket.Device
struct PyLocalSimulator <: Braket.AbstractBraketSimulator
o::Py
LocalSimulator() = new(local_sim.LocalSimulator())
LocalSimulator(backend::String) = new(local_sim.LocalSimulator(backend))
PyLocalSimulator() = new(local_sim.LocalSimulator())
PyLocalSimulator(backend::String) = new(local_sim.LocalSimulator(backend))
end
Braket.name(ls::LocalSimulator) = pyconvert(String, ls.name)

function Base.run(d::LocalSimulator, task_spec::Braket.IR.AHSProgram; shots::Int=0, kwargs...)
(ls::PyLocalSimulator)(nq::Int, shots::Int) = ls
Braket.name(ls::PyLocalSimulator) = pyconvert(String, ls.name)
Braket.device_id(ls::PyLocalSimulator) = pyconvert(String, ls._delegate.DEVICE_ID)
Braket.properties(ls::PyLocalSimulator) = ls.properties
function Braket.simulate(d::PyLocalSimulator, task_spec::Braket.IR.AHSProgram; shots::Int=0, kwargs...)
py_ir = Py(task_spec)
py_raw_result = d._delegate.run(py_ir, shots; kwargs...)
jl_raw_result = pyconvert(Braket.AnalogHamiltonianSimulationTaskResult, py_raw_result)
res = Braket.format_result(jl_raw_result)
id = res.task_metadata.id
LocalQuantumTask(id, res)
end
Base.run(d::LocalSimulator, task_spec::Braket.AnalogHamiltonianSimulation; shots::Int=0, kwargs...) = run(d, ir(task_spec); shots=shots, kwargs...)

function Base.run(d::LocalSimulator, task_spec::Braket.OpenQasmProgram; shots::Int=0, inputs::Dict{String, Float64}=Dict{String,Float64}(), kwargs...)
py_inputs = pydict(pystr(k)=>v for (k,v) in inputs)
py_ts = pyopenqasm.Program(source=pystr(task_spec.source), inputs=py_inputs)
py_raw_result = d._delegate.run_openqasm(py_ts, shots, kwargs...)
jl_result = pyconvert(Braket.GateModelTaskResult, py_raw_result)
t_id = jl_result.taskMetadata.id
res = Braket.format_result(jl_result)
LocalQuantumTask(t_id, res)
return pyconvert(Braket.AnalogHamiltonianSimulationTaskResult, py_raw_result)
end
Braket.simulate(d::PyLocalSimulator, task_spec::Braket.AnalogHamiltonianSimulation; shots::Int=0, kwargs...) = simulate(d, ir(task_spec); shots=shots, kwargs...)

function Base.run(d::LocalSimulator, task_spec::PyCircuit; shots::Int=0, inputs::Dict{String, Float64}=Dict{String,Float64}(), kwargs...)
jaqcd_ir = task_spec.to_ir(ir_type=circuit.serialization.IRType.JAQCD)
py_raw_result = d._delegate.run(jaqcd_ir, task_spec.qubit_count, shots, kwargs...)
jl_result = pyconvert(Braket.GateModelTaskResult, py_raw_result)
t_id = jl_result.taskMetadata.id
res = Braket.format_result(jl_result)
LocalQuantumTask(t_id, res)
function Braket._run_internal(simulator::PyLocalSimulator, task_spec::AnalogHamiltonianSimulation, args...; kwargs...)
raw_py_result = simulator._run_internal(Py(ir(task_spec)), args...; kwargs...)
jl_task_metadata = pyconvert(Braket.TaskMetadata, raw_py_result.task_metadata)
jl_measurements = map(raw_py_result.measurements) do m
jl_status = pyconvert(String, pystr(m.status))
status = if jl_status == "Success"
Braket.success
elseif jl_status == "Partial_success"
Braket.partial_success
else
Braket.failure
end
Braket.ShotResult(status,
pyconvert(Any, m.pre_sequence),
pyconvert(Any, m.post_sequence)
)
end
return Braket.AnalogHamiltonianSimulationQuantumTaskResult(jl_task_metadata, jl_measurements)
end
Base.run(d::LocalSimulator, task_spec::Circuit; kwargs...) = run(d, PyCircuit(task_spec); kwargs...)

(ls::LocalSimulator)(task_spec; kwargs...) = run(ls, task_spec; kwargs...)
(ls::PyLocalSimulator)(task_spec; kwargs...) = simulate(ls, task_spec; kwargs...)

Py(d::LocalSimulator) = getfield(d, :o)
Base.getproperty(d::LocalSimulator, s::Symbol) = getproperty(Py(d), s)
Base.getproperty(d::LocalSimulator, s::AbstractString) = getproperty(Py(d), s)
Base.setproperty!(d::LocalSimulator, s::Symbol, x) = setproperty!(Py(d), s, x)
Base.setproperty!(d::LocalSimulator, s::AbstractString, x) = setproperty!(Py(d), s, x)
Py(d::PyLocalSimulator) = getfield(d, :o)
Base.getproperty(d::PyLocalSimulator, s::Symbol) = getproperty(Py(d), s)
Base.getproperty(d::PyLocalSimulator, s::AbstractString) = getproperty(Py(d), s)
Base.setproperty!(d::PyLocalSimulator, s::Symbol, x) = setproperty!(Py(d), s, x)
Base.setproperty!(d::PyLocalSimulator, s::AbstractString, x) = setproperty!(Py(d), s, x)
49 changes: 0 additions & 49 deletions PyBraket/src/pycircuit.jl

This file was deleted.

84 changes: 0 additions & 84 deletions PyBraket/src/pygates.jl

This file was deleted.

Loading

0 comments on commit f357046

Please sign in to comment.