.. _qpu_example_sat_constrained:

=================================
SAT: Formulate, Embed, and Submit
=================================

The example of the :ref:`qpu_example_sat_unconstrained` section formulates an
:term:`objective function` for a simple :term:`SAT` problem. The |dwave_short|
quantum computer is also well suited to solving optimization problems with
binary variables. Real-world optimization problems often come with
:term:`constraint`\ s: conditions of the problem that any valid solution must
satisfy.

For example, when optimizing a traveling salesperson's route through a series
of cities, you need a constraint forcing the salesperson to be in exactly one
city at each stage of the trip: a solution that puts the salesperson in two or
more places at once is invalid.

.. figure:: ../_images/salesman.png
    :name: salesman
    :scale: 50 %
    :alt: Traveling salesman problem

    The traveling salesperson problem is an optimization problem that can be
    solved using exactly-one-true constraints. Map data |copy| 2017
    GeoBasis-DE/BKG (|copy| 2009), Google.

In fact, the example of the :ref:`qpu_example_sat_unconstrained` section is
actually also an example of a simple constraint, the equality (or XNOR)
constraint. The example of this section is slightly more complex. The
*exactly-one-true* constraint is the Boolean satisfiability problem of finding,
given a set of variables, when exactly one variable is TRUE (equals 1). This
example looks at an exactly-one-true constraint for three variables.

The problem-solving process is the same as described in the
:ref:`qpu_qubo_ising` section.

.. note:: Quantum samplers optimize objective functions that represent problems
    in formats that are *unconstrained* (e.g., QUBOs, or unconstrained binary
    optimization problems). As is shown in this section, any constraints in the
    original problem are represented as part of the objective function; this
    technique is known as :ref:`penalty models <concept_penalty>`.

    In addition, some of the
    `Leap service <https://cloud.dwavesys.com/leap/>`_'s quantum-classical
    hybrid solvers accept only unconstrained objective functions; for example,
    hybrid BQM solvers. Thus, for these solvers any constraints must be added
    to the objective function, typically as a penalty. However, some of the
    Leap service's hybrid solvers can handle constraints natively, as shown in
    the :ref:`opt_index_examples_beginner` examples. For more information, see
    the :ref:`opt_index_get_started` section.

.. _qpu_example_sat_constrained_formulation:

Formulating an Objective Function
=================================

For problems with a small number of binary variables (in this example three:
:math:`a`, :math:`b`, and :math:`c`), you can tabulate all the possible
permutations and identify the states when exactly one variable is 1 and the
other two are 0. You do so with a truth table:

========== ========== ========== ===================
:math:`a`  :math:`b`  :math:`c`  Exactly One is True
========== ========== ========== ===================
0          0          0          FALSE
1          0          0          TRUE
0          1          0          TRUE
1          1          0          FALSE
0          0          1          TRUE
1          0          1          FALSE
0          1          1          FALSE
1          1          1          FALSE
========== ========== ========== ===================

Notice that the three constraint-satisfying states in the truth table have in
common that the sum of variables equals :math:`1`, so you can express the
constraint mathematically as the equation,

.. math::

    a + b + c = 1.

As noted in the :ref:`qpu_qubo_ising` section, you can solve some equations by
minimization if you formulate an objective function that takes the square\ [#]_
of the subtraction of one side from another:

.. math::

    E(a,b,c) = (a + b + c - 1)^2.

.. [#]
    If you do not square, the minimum for
    :math:`\tilde{E}(a,b,c) = a + b + c - 1` is :math:`\tilde{E}(a=b=c=0) = -1`
    which is lower than for any of the three constraint-satisfying states, such
    as :math:`\tilde{E}(a=b=0;c=1)=0`.

Expanding the squared term---while remembering that for binary variables with
values 0 or 1 the square of a variable is itself, :math:`X^2 = X`---shows in
explicit form the objective function's quadratic and linear terms.

.. math::

    E(a,b,c) &= a^2 + ab + ac -a + ba + b^2 + bc -b + ca + cb + c^2
    - c - a - b - c +1 \\
    &= a^2 + b^2 + c^2 + 2ab + 2ac + 2bc - 2a - 2b - 2c + 1 \\
    &= 2ab + 2ac + 2bc - a - b - c + 1,

Notice that this objective formula matches the
:ref:`QUBO format <qpu_qubo_ising>` for three variables,

.. math::

        E_{qubo}(a_i, b_{i,j}; q_i) &= \sum_{i} a_i q_i +
        \sum_{i<j} b_{i,j} q_i q_j \\
        E_{qubo}(a_i, b_{i,j}; q_1, q_2, q_3) &= a_1 q_1 + a_2 q_2 + a_3 q_3 +
        b_{1,2} q_1 q_2 + b_{1,3} q_1 q_3 + b_{2,3} q_2 q_3 ,

where :math:`a_i=-1` and :math:`b_{i,j}=2`, with a difference of the :math:`+1`
term.\ [#]_

.. [#]
    A constant term in an objective function does not affect the solutions
    because it just increases or decreases energies (values of the objective)
    for all states by the same amount, preserving relative ordering.

Below, the truth table is shown with an additional column of the energy for
the objective function found above. The lowest energy states (best solutions)
are those that match the exactly-one-true constraint.

========== ========== ========== ==================== ==============
:math:`a`  :math:`b`  :math:`c`  Exactly One is True  Energy
========== ========== ========== ==================== ==============
0          0          0          FALSE                1
1          0          0          TRUE                 0
0          1          0          TRUE                 0
1          1          0          FALSE                1
0          0          1          TRUE                 0
1          0          1          FALSE                1
0          1          1          FALSE                1
1          1          1          FALSE                4
========== ========== ========== ==================== ==============

Clearly, a solver minimizing the objective function
:math:`2ab + 2ac + 2bc - a - b - c` can be expected to return solutions (values
of variables :math:`a, b, c`) that satisfy the original problem of an
*exactly-one-true* constraint.

As explained in the :ref:`qpu_example_sat_unconstrained` example, to solve a
QUBO with a |dwave_short| quantum computer, you must map (minor embed) it to the
QPU. That step is explained in detail in the next section.

Minor Embedding
===============

This section explains how the QUBO created in the previous section is
minor-embedded onto a QPU, in this case, an |dwave_5kq| QPU with its
:ref:`Pegasus <qpu_topologies>` graph.

|dwave_short| provides automatic minor-embedding tools, and if you are
submitting your problem to a
`Leap service <https://cloud.dwavesys.com/leap/>`_'s quantum-classical hybrid
solver, the solver handles all interactions with the QPU.

The QUBO developed for an exactly-one-true constraint with three variables in
the previous section, :math:`2ab + 2ac + 2bc - a - b - c`, can be represented by
the triangular graph shown in :numref:`Figure %s <triangle>`.

.. figure:: ../_images/triangle.png
    :name: triangle
    :scale: 50 %
    :alt: Triangular graph

    Triangular graph for an exactly-one-true constraint with its biased nodes
    and edges.

As explained in the :ref:`qpu_embedding_intro` section, nodes that represent the
objective function's variables such as :math:`a` are mapped to :term:`qubits` on
the QPU while edges that represent the objective function's quadratic terms such
as :math:`ab` are mapped to :term:`couplers`.

:numref:`Figure %s <triangleEmbeddingPegasus>` shows such a mapping, between
the graph representing the QUBO on the left and one particular minor-embedding
on the right. (Rerunning Ocean software's :ref:`minorminer <index_minorminer>`
tool, which produced this minor embedding, generates embeddings to various
qubits across the QPU; the particular qubit numbers noted here are unimportant.)

*   Nodes :math:`a, b, c` (grey circles in the left-hand panel) map to qubits
    :math:`1812, 5169, 1827` (blue circles in the right-hand panel),
    respectively.
*   Edges :math:`ab, bc, ca` (orange lines in the left-hand panel) map to
    couplers :math:`[1812, 5619], [1827, 5619], [1812, 1827]` (blue lines in
    the right-hand panel), respectively.

.. figure:: ../_images/triangle_embedding_pegasus.png
    :name: triangleEmbeddingPegasus
    :scale: 50 %
    :alt: Embedding in the Pegasus topology

    Embedding in the Pegasus topology for an exactly-one-true constraint
    rendered by Ocean software's problem inspector. The original QUBO is
    represented on the left and its embedded representation on the right.

But, as the :ref:`qpu_topologies` section notes, |dwave_short| QPUs are not
fully connected. For larger graphs than the example above, you may not always be
able to map each node to a qubit and find connecting couplers to represent all
edges.

How are more complex graphs minor-embedded? Minor embedding often requires
:term:`chains`.

.. _qpu_example_sat_constrained_chains:

Chains
------

To understand how chaining qubits overcomes the problem of sparse connectivity,
consider minor embedding the triangular graph of :numref:`Figure %s <triangle>`
into two target graphs, one sparser than the other.
:numref:`Figure %s <chainTriangleFourQubits>` shows two such embeddings:
the triangular graph is mapped on the left to a fully-connected graph of four
nodes (called a :math:`K_4` complete graph) and on the right to a sparser graph,
also of four nodes.
For the left-hand embedding, you can choose any mapping between :math:`a, b, c`
and :math:`0, 1, 2, 3`; here :math:`a, b, c` are mapped to :math:`2, 0, 1`,
respectively. For the right-hand embedding, however, no choice of just three
target nodes suffices. The same :math:`2, 0, 1` target nodes leaves :math:`b`
disconnected from :math:`c`. Chaining target nodes :math:`0` and :math:`3` to
represent node :math:`b` makes use of both the connection between :math:`0` to
:math:`2` and the connection between :math:`3` and :math:`1`.

.. figure:: ../_images/chain_triangle_four_qubits.png
    :name: chainTriangleFourQubits
    :alt: Embedding a triangular graph into the Chimera graph by using a chain.

    Embedding a triangular graph into fully connected and sparse four-node
    graphs.

On QPUs, chaining qubits is accomplished by setting the strength of their
connecting couplers negative enough to strongly correlate the states of the
chained qubits; if at the end of most anneals these qubits are in the same
classical state, representing the same binary value in the objective function,
they are in effect acting as a single variable.

As an example, consider a fully-connected graph of five nodes (a :math:`K_5`
graph). Such a graph cannot be mapped to five qubits of an |dwave_5kq| QPU
because the Pegasus graph's connectivity is too sparse. Instead, some nodes are
mapped to chains of qubits.

:numref:`Figure %s <embeddingK5Pegasus>` shows a :math:`K_5` graph of some
arbitrary problem on the left and a minor-embedding on the right. Here,
variable 3 (highlighted magenta) is represented by a two-qubit chain of qubits
4408 and 2437 (highlighted magenta) while variables 0, 1, 2, and 4 are
represented by single qubits 4333, 4348, 2497, and 2512.

.. figure:: ../_images/embedding_k5_pegasus.png
    :name: embeddingK5Pegasus
    :scale: 50 %
    :alt: Embedding of a K5 graph in the Pegasus topology

    Embedding for a :math:`K_5` fully connected graph in the Pegasus topology
    rendered by Ocean software's problem inspector. The original QUBO is
    represented on the left and its embedded representation, with its two-qubit
    chain, on the right.

.. _qpu_example_sat_constrained_manual_embedding:

Manual Minor-Embedding
----------------------

Manually minor-embedding a problem is typically undertaken only for problems
that have either very few variables or a repetitive structure that maps to unit
cells of the QPU topology---in both cases you work with one or more unit cells.
Additionally, you might make minor adjustments to an embedding found by
software. You are unlikely to manually embed a random 100-variable problem.

This section provides an example of how you can calculate the biases needed for
minor-embedding on a simple problem. Ocean software's minor-embedding tools,
such as :ref:`minorminer <index_minorminer>`, do similar calculations.

.. dropdown:: Example of Manual Minor Embedding

    This example returns to the QUBO developed for an exactly-one-true
    constraint with three variables in the
    :ref:`qpu_example_sat_constrained_formulation` section above,
    :math:`2ab + 2ac + 2bc - a - b - c`, as represented by the triangular graph
    shown in :numref:`Figure %s <triangle>` above. For simplicity, it is
    minor-embedded into a :ref:`Chimera <topology_intro_chimera>` graph.

    To see how a triangular graph fits on the Chimera graph, take a closer look
    at the unit cell in the Chimera topology shown in
    :numref:`Figure %s <unit-cell>`. Notice that there is no way to make a
    triangular closed loop of three qubits and their connecting edges. However,
    you can make a closed loop of four qubits and their edges using, say,
    qubits 0, 1, 4, and 5.

    .. figure:: ../_images/unit-cell.png
        :name: unit-cell
        :scale: 35 %
        :alt: Unit cell

        Unit cell in the Chimera topology.

    As in the example of the :ref:`qpu_example_sat_constrained_chains` section,
    make a three-node loop of a four-node structure by representing a single
    variable with a chain of two qubits. :numref:`Figure %s <embedding-gs>`
    shows a chaining of qubit 0 and qubit 5 to represent variable :math:`b`.

    .. figure:: ../_images/embedding.png
        :name: embedding-gs
        :alt: Embedding a triangular graph into the Chimera graph by using a
            chain.

        Embedding a triangular graph into the Chimera graph by using a chain.

    Here, for qubits 0 and 5 to represent variable :math:`b`, the strength of
    the coupler between them must be set negative enough.

    The mapping of :numref:`Figure %s <embedding-gs>` is straightforward for
    non-chained qubits with biases being the linear coefficients of the
    objective function and coupler strengths the quadratic coefficients:

    *   Variables :math:`a` and :math:`c`, represented by qubits 4 and 1,
        respectively, have bias :math:`-1`.
    *   Edges :math:`(a,b), (a,c), (b,c)`, represented by couplers
        :math:`(0,4), (1,4), (1,5)`, respectively, have strengths :math:`2`.

    To chain qubits 0 and 5 to represent variable :math:`b` requires that you
    add a strong negative coupling strength between them. This coupling has no
    corresponding quadratic coefficient in the objective function, so other
    biases must be adjusted to compensate. This process requires a few steps:

    1.  Evenly split the bias of :math:`-1` from variable :math:`b` between
        qubits 0 and 5. Now the bias of these two qubits is :math:`-0.5`.
    2.  Choose a strong negative coupling strength for the chain between qubits
        0 and 5. This example arbitrarily chooses :math:`-3` because it is
        stronger than the values for couplers around it.\ [#]_
    3.  Compensate for the :math:`-3` added in step 2 by adding
        :math:`-\frac{-3}{2} = 1.5` to each bias of qubits 0 and 5. Now the
        biases for these qubits are :math:`1`.

    .. [#]

        Setting chain strengths is further discussed in the
        :ref:`qpu_basic_config_chain_strength` section.

    The resulting minor-embedding values are shown in the tables below.

    .. table:: Minor Embedding: Linear Coefficients.

        +------+-------------+--------+------+
        |      | Linear      |        |      |
        | Node | Coefficient | Qubits | Bias |
        +======+=============+========+======+
        | a    | -1          | 4      | -1   |
        +------+-------------+--------+------+
        | b    | -1          | 0, 5   | 1, 1 |
        +------+-------------+--------+------+
        | c    | -1          | 1      | -1   |
        +------+-------------+--------+------+

    .. table:: Minor Embedding: Quadratic Coefficients.

        +-------+-------------+---------+----------+
        |       | Quadratic   |         |          |
        | Edge  | Coefficient | Coupler | Strength |
        +=======+=============+=========+==========+
        | (a,b) | 2           | (0,4)   | 2        |
        +-------+-------------+---------+----------+
        | (a,c) | 2           | (1,4)   | 2        |
        +-------+-------------+---------+----------+
        | (b,c) | 2           | (1,5)   | 2        |
        +-------+-------------+---------+----------+
        |       |             | (0,5)   | -3       |
        +-------+-------------+---------+----------+

    You program the quantum computer to solve this problem by configuring the
    QPU's qubits with these biases and its couplers with these strengths.

    .. note::

        When using the QUBO formulation, as in this example, you compensate for
        the quadratic term a chain introduces into the objective by adding its
        negative, divided by the number of qubits in the chain, to the biases
        of the chain's qubits; this compensation is not used for the Ising
        formulation, where the the energies of valid solutions are simply
        shifted by the introduced quadratic term.

    The solutions returned from the QPU express the states of qubits at the end
    of each anneal. To translate qubit states to values of the problem
    variables, the solutions must be *unembedded*.

    For example, consider the following results for 1000 anneals:

    +----------+---+---+---+---+----------------+
    |  Energy  |      Qubit    | Occurrences    |
    +          +---+---+---+---+                +
    |          | 0 | 5 | 4 | 1 |                |
    +==========+===+===+===+===+================+
    | -1.0     | 0 | 0 | 1 | 0 |  206           |
    +----------+---+---+---+---+----------------+
    | -1.0     | 0 | 0 | 0 | 1 |  526           |
    +----------+---+---+---+---+----------------+
    | -1.0     | 1 | 1 | 0 | 0 |  267           |
    +----------+---+---+---+---+----------------+
    | 0.0      | 1 | 1 | 0 | 1 |  1             |
    +----------+---+---+---+---+----------------+

    For this simple example with its single chain, unembedding consists of
    mapping qubits 4, 1 to variables :math:`a, c`, and qubits 0, 5 to variable
    :math:`b`. The results in the table above unembedded to:

    *   Row 1: Solution :math:`(a, b, c) = (1, 0, 0)` with energy :math:`-1`
        was found 206 times.
    *   Row 2: Solution :math:`(a, b, c) = (0, 0, 1)` with energy :math:`-1`
        was found 526 times.
    *   Row 3: Solution :math:`(a, b, c) = (0, 1, 0)` with energy :math:`-1`
        was found 267 times.

    One anneal ended with result :math:`(a, b, c) = (0, 1, 1)`, which is not a
    correct solution, and has a higher energy than the correct solutions.

    .. note::

        Notice also that the energy of the valid solutions, the ground-state
        energy, is :math:`-1`, not the zero calculated in the
        :ref:`qpu_example_sat_constrained_formulation` section's truth table.
        This is because of the constant :math:`+1` dropped from the objective
        function, :math:`E(a,b,c) = 2ab + 2ac + 2bc - a - b - c + 1`.

The next section shows how you submit a problem to a |dwave_short| quantum
computer.

Submitting
==========

This section uses |dwave_short|'s open-source :ref:`Ocean SDK <index_ocean_sdk>`
to submit the exactly-one-true problem formulated in the previous subsections.

Before you can submit a problem to |dwave_short| solvers, you must have an
account and an API token; visit the
`Leap service <https://cloud.dwavesys.com/leap/>`_ to sign up for an account
and get your token.

.. note::
    To run the following steps yourself requires prior configuration of some
    requisite information for problem submission through SAPI. If you have
    installed the Ocean SDK or are using a :ref:`supported IDE <leap_dev_env>`,
    this is typically done as a first step.

For more information, including on Ocean SDK installation instructions and
detailed examples, see the
:ref:`Ocean software documentation <index_ocean_sdk>`.

The Ocean software can heuristically find minor-embeddings for your QUBO or
Ising objective functions, as shown here.

First, select a quantum computer. Ocean software provides
:meth:`feature-based solver selection <dwave.cloud.client.Client.get_solvers>`,
enabling you to select a quantum computer that meets your requirements on its
number of qubits, topology, particular features, etc. This example, uses the
default.

.. testcode::

    from dwave.system import DWaveSampler, EmbeddingComposite
    sampler = EmbeddingComposite(DWaveSampler())

Set values of the QUBO and submit to the selected QPU.

.. testcode::

    linear = {('a', 'a'): -1, ('b', 'b'): -1, ('c', 'c'): -1}
    quadratic = {('a', 'b'): 2, ('b', 'c'): 2, ('a', 'c'): 2}
    Q = {**linear, **quadratic}

    sampleset = sampler.sample_qubo(Q, num_reads=5000)

Below are results from running this problem on a |dwave_5kq| system:

>>> print(sampleset)                        # doctest: +SKIP
   a  b  c energy num_oc. chain_b.
0  0  0  1   -1.0    1591     0.0
1  1  0  0   -1.0    2040     0.0
2  0  1  0   -1.0    1365     0.0
3  1  0  1    0.0       2     0.0
4  1  1  0    0.0       1     0.0
5  0  1  1    0.0       1     0.0
['BINARY', 6 rows, 5000 samples, 3 variables]

In the results of 5000 reads, you see that the lowest energy occurs for the
three valid solutions to the problem.