From 9fbd1bf6b426b533bb4bbf2c6f1f198f69440739 Mon Sep 17 00:00:00 2001 From: Wong Zi Cheng <70616433+chmwzc@users.noreply.github.com> Date: Tue, 2 Jan 2024 08:25:46 +0000 Subject: [PATCH 01/11] Uncomment and fix bug in ucc_ansatz function --- src/qibochem/ansatz/ucc.py | 163 ++++++++++++++++++++----------------- 1 file changed, 90 insertions(+), 73 deletions(-) diff --git a/src/qibochem/ansatz/ucc.py b/src/qibochem/ansatz/ucc.py index 8b3bffd..b1b08b0 100644 --- a/src/qibochem/ansatz/ucc.py +++ b/src/qibochem/ansatz/ucc.py @@ -6,6 +6,8 @@ import openfermion from qibo import gates, models +from qibochem.ansatz.hf_reference import hf_circuit + def expi_pauli(n_qubits, theta, pauli_string): """ @@ -166,9 +168,9 @@ def generate_excitations(order, excite_from, excite_to, conserve_spin=True): if order > min(len(excite_from), len(excite_to)): return [[]] + # Generate all possible excitations first from itertools import combinations - # Generate all possible excitations first all_excitations = [ [*_from, *_to] for _from in combinations(excite_from, order) for _to in combinations(excite_to, order) ] @@ -228,7 +230,7 @@ def sort_excitations(excitations): pair_excitations = sorted(pair_excitations) ex_to_remove = pair_excitations.pop(0) if ex_to_remove in copy_excitations: - # 'Move' the first entry of same_mo_index from copy_excitations to result + # 'Move' the first pair excitation from copy_excitations to result index = copy_excitations.index(ex_to_remove) result.append(copy_excitations.pop(index)) @@ -253,77 +255,92 @@ def sort_excitations(excitations): result.append(copy_excitations.pop(index)) prev = None continue - # Remove the first entry from the sorted list of remaining excitations and add it to result - prev = copy_excitations.pop(0) - result.append(prev) + if copy_excitations: + # Remove the first entry from the sorted list of remaining excitations and add it to result + prev = copy_excitations.pop(0) + result.append(prev) return result -# def ucc_ansatz(molecule, excitation_level=None, excitations=None, thetas=None, trotter_steps=1, ferm_qubit_map=None): -# """ -# Build a circuit corresponding to the UCC ansatz with multiple excitations for a given Molecule. -# If no excitations are given, it defaults to returning the full UCCSD circuit ansatz for the -# Molecule. -# -# Args: -# molecule: Molecule of interest. -# excitation_level: Include excitations up to how many electrons, i.e. ("S", "D", "T", "Q") -# Ignored if `excitations` argument is given. Default is "D", i.e. double excitations -# excitations: List of excitations (e.g. `[[0, 1, 2, 3], [0, 1, 4, 5]]`) used to build the -# UCC circuit. Overrides the `excitation_level` argument -# thetas: Parameters for the excitations. Defaults to an array of zeros if not given -# trotter_steps: number of Trotter steps; i.e. number of times the UCC ansatz is applied with -# theta=theta/trotter_steps. Default is 1 -# ferm_qubit_map: fermion-to-qubit transformation. Default is Jordan-Wigner ("jw") -# -# Returns: -# circuit: Qibo Circuit corresponding to the UCC ansatz -# """ -# # Get the number of electrons and virtual orbitals from the molecule argument -# n_elec = sum(molecule.nelec) if molecule.n_active_e is None else molecule.n_active_e -# n_orbs = molecule.nso if molecule.n_active_orbs is None else molecule.n_active_orbs -# -# # Define the excitation level to be used if no excitations given -# if excitations is None: -# excitation_levels = ("S", "D", "T", "Q") -# if excitation_level is None: -# excitation_level = "D" -# else: -# # Check validity of input -# assert len(excitation_level) == 1 and excitation_level.upper() in excitation_levels -# # Note: Probably don't be too ambitious and try to do 'T'/'Q' at the moment... -# if excitation_level.upper() in ("T", "Q"): -# raise NotImplementedError("Cannot handle triple and quadruple excitations!") -# # Get the (largest) order of excitation to use -# excitation_order = excitation_levels.index(excitation_level.upper()) + 1 -# -# # Generate and sort all the possible excitations -# excitations = [] -# for order in range(excitation_order, 0, -1): # Reversed to get higher excitations first -# excitations += sort_excitations(generate_excitations(order, range(0, n_elec), range(n_elec, n_orbs))) -# else: -# # Some checks to ensure the given excitations are valid -# assert all(len(_ex) % 2 == 0 for _ex in excitations), "Excitation with an odd number of elements found!" -# -# # Check if thetas argument given, define to be all zeros if not -# # TODO: Unsure if want to use MP2 guess amplitudes for the doubles? Some say good, some say bad -# # Number of circuit parameters: S->2, D->8, (T/Q->32/128; Not sure?) -# n_parameters = 2 * len([_ex for _ex in excitations if len(_ex) == 2]) # Singles -# n_parameters += 8 * len([_ex for _ex in excitations if len(_ex) == 4]) # Doubles -# if thetas is None: -# thetas = np.zeros(n_parameters) -# else: -# # Check that number of circuit variables (i.e. thetas) matches the number of circuit parameters -# assert len(thetas) == n_parameters, "Number of input parameters doesn't match the number of circuit parameters!" -# -# # Build the circuit -# circuit = models.Circuit(n_orbs) -# for excitation, theta in zip(excitations, thetas): -# # coeffs = [] -# circuit += ucc_circuit( -# n_orbs, excitation, theta, trotter_steps=trotter_steps, ferm_qubit_map=ferm_qubit_map # , coeffs=coeffs) -# ) -# # if isinstance(all_coeffs, list): -# # all_coeffs.append(np.array(coeffs)) -# -# return circuit +def ucc_ansatz( + molecule, + excitation_level=None, + excitations=None, + thetas=None, + trotter_steps=1, + ferm_qubit_map=None, + include_hf=True, +): + """ + Build a circuit corresponding to the UCC ansatz with multiple excitations for a given Molecule. + If no excitations are given, it defaults to returning the full UCCSD circuit ansatz for the + Molecule. + + Args: + molecule: Molecule of interest. + excitation_level: Include excitations up to how many electrons, i.e. ("S", "D", "T", "Q") + Ignored if `excitations` argument is given. Default is "D", i.e. double excitations + excitations: List of excitations (e.g. `[[0, 1, 2, 3], [0, 1, 4, 5]]`) used to build the + UCC circuit. Overrides the `excitation_level` argument + thetas: Parameters for the excitations. Defaults to an array of zeros if not given + trotter_steps: number of Trotter steps; i.e. number of times the UCC ansatz is applied with + theta=theta/trotter_steps. Default is 1 + ferm_qubit_map: fermion-to-qubit transformation. Default: Jordan-Wigner ("jw") + include_hf: Whether or not to start the circuit with a Hartree-Fock circuit. Default: ``True`` + + Returns: + circuit: Qibo Circuit corresponding to the UCC ansatz + """ + # Get the number of electrons and spin-orbitals from the molecule argument + n_elec = sum(molecule.nelec) if molecule.n_active_e is None else molecule.n_active_e + n_orbs = molecule.nso if molecule.n_active_orbs is None else molecule.n_active_orbs + + # Define the excitation level to be used if no excitations given + if excitations is None: + excitation_levels = ("S", "D", "T", "Q") + if excitation_level is None: + excitation_level = "D" + else: + # Check validity of input + assert ( + len(excitation_level) == 1 and excitation_level.upper() in excitation_levels + ), "Unknown input for excitation_level" + # Note: Probably don't be too ambitious and try to do 'T'/'Q' at the moment... + if excitation_level.upper() in ("T", "Q"): + raise NotImplementedError("Cannot handle triple and quadruple excitations!") + # Get the (largest) order of excitation to use + excitation_order = excitation_levels.index(excitation_level.upper()) + 1 + + # Generate and sort all the possible excitations + excitations = [] + for order in range(excitation_order, 0, -1): # Reversed to get higher excitations first + excitations += sort_excitations(generate_excitations(order, range(0, n_elec), range(n_elec, n_orbs))) + else: + # Some checks to ensure the given excitations are valid + assert all(len(_ex) % 2 == 0 for _ex in excitations), "Excitation with an odd number of elements found!" + + # Check if thetas argument given, define to be all zeros if not + # TODO: Unsure if want to use MP2 guess amplitudes for the doubles? Some say good, some say bad + # Number of circuit parameters: S->2, D->8, (T/Q->32/128; Not sure?) + n_parameters = 2 * len([_ex for _ex in excitations if len(_ex) == 2]) # Singles + n_parameters += 8 * len([_ex for _ex in excitations if len(_ex) == 4]) # Doubles + if thetas is None: + thetas = np.zeros(n_parameters) + else: + # Check that number of circuit variables (i.e. thetas) matches the number of circuit parameters + assert len(thetas) == n_parameters, "Number of input parameters doesn't match the number of circuit parameters!" + + # Build the circuit + if include_hf: + circuit = hf_circuit(n_orbs, n_elec, ferm_qubit_map=ferm_qubit_map) + else: + circuit = models.Circuit(n_orbs) + for excitation, theta in zip(excitations, thetas): + # coeffs = [] + circuit += ucc_circuit( + n_orbs, excitation, theta, trotter_steps=trotter_steps, ferm_qubit_map=ferm_qubit_map # , coeffs=coeffs) + ) + # if isinstance(all_coeffs, list): + # all_coeffs.append(np.array(coeffs)) + + return circuit From 2ca730b0e2bf73cd555a1453eb3efdc6360f1f57 Mon Sep 17 00:00:00 2001 From: Wong Zi Cheng <70616433+chmwzc@users.noreply.github.com> Date: Wed, 3 Jan 2024 03:04:33 +0000 Subject: [PATCH 02/11] Extend MP2 guess amplitude function for singles Aka:just return 0.0 if input excitation is a single excitation --- src/qibochem/ansatz/ucc.py | 24 +++++++++++++++--------- 1 file changed, 15 insertions(+), 9 deletions(-) diff --git a/src/qibochem/ansatz/ucc.py b/src/qibochem/ansatz/ucc.py index b1b08b0..2e7711f 100644 --- a/src/qibochem/ansatz/ucc.py +++ b/src/qibochem/ansatz/ucc.py @@ -118,22 +118,28 @@ def ucc_circuit(n_qubits, excitation, theta=0.0, trotter_steps=1, ferm_qubit_map # Utility functions -def mp2_amplitude(orbitals, orbital_energies, tei): +def mp2_amplitude(excitation, orbital_energies, tei): """ - Calculate the MP2 guess amplitudes to be used in the UCC doubles ansatz - In SO basis: t_{ij}^{ab} = (g_{ijab} - g_{ijba}) / (e_i + e_j - e_a - e_b) + Calculate the MP2 guess amplitude for a single UCC circuit: 0.0 for a single excitation. + for a double excitation (In SO basis): t_{ij}^{ab} = (g_{ijab} - g_{ijba}) / (e_i + e_j - e_a - e_b) Args: - orbitals: list of spin-orbitals representing a double excitation, must have exactly - 4 elements + excitation: Iterable of spin-orbitals representing a excitation. Must have either 2 or 4 elements exactly, + representing a single or double excitation respectively. orbital_energies: eigenvalues of the Fock operator, i.e. orbital energies tei: Two-electron integrals in MO basis and second quantization notation + + Returns: + MP2 guess amplitude (float) """ - # Checks orbitals - assert len(orbitals) == 4, f"{orbitals} must have only 4 orbitals for a double excitation" - # Convert orbital indices to be in MO basis - mo_orbitals = [_orb // 2 for _orb in orbitals] + # Checks validity of excitation argument + assert len(orbitals) // 2 == 0 and len(orbitals) // 2 <= 2, f"{excitation} must have either 2 or 4 elements" + # If single excitation, can just return 0.0 directly + if len(orbitals) == 2: + return 0.0 + # Convert orbital indices to be in MO basis + mo_orbitals = [orbitalin // 2 for orbital in excitation] # Numerator: g_ijab - g_ijba g_ijab = ( tei[tuple(mo_orbitals)] # Can index directly using the MO TEIs From 6799e094b57ca0c58e14f5fae727b97608054d7a Mon Sep 17 00:00:00 2001 From: Wong Zi Cheng <70616433+chmwzc@users.noreply.github.com> Date: Wed, 3 Jan 2024 03:18:18 +0000 Subject: [PATCH 03/11] Fix minor bug --- src/qibochem/ansatz/ucc.py | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/src/qibochem/ansatz/ucc.py b/src/qibochem/ansatz/ucc.py index 2e7711f..4dcd7ad 100644 --- a/src/qibochem/ansatz/ucc.py +++ b/src/qibochem/ansatz/ucc.py @@ -133,22 +133,22 @@ def mp2_amplitude(excitation, orbital_energies, tei): MP2 guess amplitude (float) """ # Checks validity of excitation argument - assert len(orbitals) // 2 == 0 and len(orbitals) // 2 <= 2, f"{excitation} must have either 2 or 4 elements" + assert len(excitation) // 2 == 0 and len(excitation) // 2 <= 2, f"{excitation} must have either 2 or 4 elements" # If single excitation, can just return 0.0 directly - if len(orbitals) == 2: + if len(excitation) == 2: return 0.0 # Convert orbital indices to be in MO basis - mo_orbitals = [orbitalin // 2 for orbital in excitation] + mo_orbitals = [orbital // 2 for orbital in excitation] # Numerator: g_ijab - g_ijba g_ijab = ( tei[tuple(mo_orbitals)] # Can index directly using the MO TEIs - if (orbitals[0] + orbitals[3]) % 2 == 0 and (orbitals[1] + orbitals[2]) % 2 == 0 + if (excitation[0] + excitation[3]) % 2 == 0 and (excitation[1] + excitation[2]) % 2 == 0 else 0.0 ) g_ijba = ( tei[tuple(mo_orbitals[:2] + mo_orbitals[2:][::-1])] # Reverse last two terms - if (orbitals[0] + orbitals[2]) % 2 == 0 and (orbitals[1] + orbitals[3]) % 2 == 0 + if (excitation[0] + excitation[2]) % 2 == 0 and (excitation[1] + excitation[3]) % 2 == 0 else 0.0 ) numerator = g_ijab - g_ijba From 03043f663bdf667bb249ff7e82f2e2b3f5d73a9e Mon Sep 17 00:00:00 2001 From: Wong Zi Cheng <70616433+chmwzc@users.noreply.github.com> Date: Wed, 3 Jan 2024 03:29:03 +0000 Subject: [PATCH 04/11] Fix another stupid bug Haiz, first time working in office in 2024, somehow the mind is sooooo sloppy and careless --- src/qibochem/ansatz/ucc.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/qibochem/ansatz/ucc.py b/src/qibochem/ansatz/ucc.py index 4dcd7ad..aca17d4 100644 --- a/src/qibochem/ansatz/ucc.py +++ b/src/qibochem/ansatz/ucc.py @@ -133,7 +133,7 @@ def mp2_amplitude(excitation, orbital_energies, tei): MP2 guess amplitude (float) """ # Checks validity of excitation argument - assert len(excitation) // 2 == 0 and len(excitation) // 2 <= 2, f"{excitation} must have either 2 or 4 elements" + assert len(excitation) % 2 == 0 and len(excitation) // 2 <= 2, f"{excitation} must have either 2 or 4 elements" # If single excitation, can just return 0.0 directly if len(excitation) == 2: return 0.0 From 3721e10301486439492a962e8e1bd0c3493e27c8 Mon Sep 17 00:00:00 2001 From: Wong Zi Cheng <70616433+chmwzc@users.noreply.github.com> Date: Wed, 3 Jan 2024 06:31:21 +0000 Subject: [PATCH 05/11] Edit and expand UCC test cases --- tests/test_ucc.py | 180 +++++++++++++++++++++++++++++----------------- 1 file changed, 115 insertions(+), 65 deletions(-) diff --git a/tests/test_ucc.py b/tests/test_ucc.py index 46b50c5..1c759e1 100644 --- a/tests/test_ucc.py +++ b/tests/test_ucc.py @@ -1,3 +1,5 @@ +from functools import partial + import numpy as np import pytest from qibo import gates @@ -8,6 +10,7 @@ generate_excitations, mp2_amplitude, sort_excitations, + ucc_ansatz, ucc_circuit, ) from qibochem.driver.molecule import Molecule @@ -76,7 +79,11 @@ def test_sort_excitations_3(): sort_excitations([[1, 2, 3, 4, 5, 6]]) -def test_mp2_amplitude(): +def test_mp2_amplitude_singles(): + assert mp2_amplitude([0, 2], np.random.rand(4), np.random.rand(4, 4)) == 0.0 + + +def test_mp2_amplitude_doubles(): h2 = Molecule([("H", (0.0, 0.0, 0.0)), ("H", (0.0, 0.0, 0.7))]) h2.run_pyscf() l = mp2_amplitude([0, 1, 2, 3], h2.eps, h2.tei) @@ -85,7 +92,113 @@ def test_mp2_amplitude(): assert np.isclose(l, ref_l) -def test_uccsd(): +def test_ucc_jw_singles(): + """Build a UCC singles (JW) circuit""" + rx_gate = partial(gates.RX, theta=-0.5 * np.pi, trainable=False) + cnot_cascade = [gates.CNOT(2, 1), gates.CNOT(1, 0), gates.RZ(0, 0.0), gates.CNOT(1, 0), gates.CNOT(2, 1)] + basis_rotation_gates = ([rx_gate(0), gates.H(2)], [gates.H(0), rx_gate(2)]) + + # Build control list of gates + nested_gate_list = [gate_list + cnot_cascade + gate_list for gate_list in basis_rotation_gates] + gate_list = [_gate for gate_list in nested_gate_list for _gate in gate_list] + + # Test ucc_function + circuit = ucc_circuit(4, [0, 2], ferm_qubit_map="jw") + # Check gates are correct + assert all( + control.name == test.name and control.target_qubits == test.target_qubits + for control, test in zip(gate_list, list(circuit.queue)) + ) + # Check that only two parametrised gates + assert len(circuit.get_parameters()) == 2 + + +def test_ucc_jw_doubles(): + """Build a UCC doubles (JW) circuit""" + rx_gate = partial(gates.RX, theta=-0.5 * np.pi, trainable=False) + cnot_cascade = [ + gates.CNOT(3, 2), + gates.CNOT(2, 1), + gates.CNOT(1, 0), + gates.RZ(0, 0.0), + gates.CNOT(1, 0), + gates.CNOT(2, 1), + gates.CNOT(3, 2), + ] + basis_rotation_gates = ( + [gates.H(0), gates.H(1), rx_gate(2), gates.H(3)], + [rx_gate(0), rx_gate(1), rx_gate(2), gates.H(3)], + [rx_gate(0), gates.H(1), gates.H(2), gates.H(3)], + [gates.H(0), rx_gate(1), gates.H(2), gates.H(3)], + [rx_gate(0), gates.H(1), rx_gate(2), rx_gate(3)], + [gates.H(0), rx_gate(1), rx_gate(2), rx_gate(3)], + [gates.H(0), gates.H(1), gates.H(2), rx_gate(3)], + [rx_gate(0), rx_gate(1), gates.H(2), rx_gate(3)], + ) + + # Build control list of gates + nested_gate_list = [gate_list + cnot_cascade + gate_list for gate_list in basis_rotation_gates] + gate_list = [_gate for gate_list in nested_gate_list for _gate in gate_list] + + # Test ucc_function + circuit = ucc_circuit(4, [0, 1, 2, 3], ferm_qubit_map="jw") + # Check gates are correct + assert all( + control.name == test.name and control.target_qubits == test.target_qubits + for control, test in zip(gate_list, list(circuit.queue)) + ) + # Check that only two parametrised gates + assert len(circuit.get_parameters()) == 8 + + +def test_ucc_bk_singles(): + """Build a UCC doubles (BK) circuit""" + rx_gate = partial(gates.RX, theta=-0.5 * np.pi, trainable=False) + cnot_cascade = [gates.CNOT(2, 1), gates.CNOT(1, 0), gates.RZ(0, 0.0), gates.CNOT(1, 0), gates.CNOT(2, 1)] + basis_rotation_gates = ([gates.H(0), rx_gate(1), gates.H(2)], [rx_gate(0), rx_gate(1), rx_gate(2)]) + + # Build control list of gates + nested_gate_list = [gate_list + cnot_cascade + gate_list for gate_list in basis_rotation_gates] + gate_list = [_gate for gate_list in nested_gate_list for _gate in gate_list] + + # Test ucc_function + circuit = ucc_circuit(4, [0, 2], ferm_qubit_map="bk") + # Check gates are correct + assert all( + control.name == test.name and control.target_qubits == test.target_qubits + for control, test in zip(gate_list, list(circuit.queue)) + ) + # Check that only two parametrised gates + assert len(circuit.get_parameters()) == 2 + + +def test_ucc_ferm_qubit_map_error(): + """If unknown fermion to qubit map used""" + with pytest.raises(KeyError): + ucc_circuit(2, [0, 1], ferm_qubit_map="Unknown") + + +def test_ucc_ansatz_h2(): + """Test the default arguments of ucc_ansatz using H2""" + mol = Molecule([("H", (0.0, 0.0, 0.0)), ("H", (0.0, 0.0, 0.7))]) + mol.run_pyscf() + + # Build control circuit + control_circuit = hf_circuit(4, 2) + for excitation in ([0, 1, 2, 3], [0, 2], [1, 3]): + control_circuit += ucc_circuit(4, excitation) + + test_circuit = ucc_ansatz(mol) + + assert all( + control.name == test.name and control.target_qubits == test.target_qubits + for control, test in zip(list(control_circuit.queue), list(test_circuit.queue)) + ) + # Check that number of parametrised gates is the same + assert len(control_circuit.get_parameters()) == len(test_circuit.get_parameters()) + + +def test_ucc_ansatz_h2(): # Define molecule and populate mol = Molecule([("Li", (0.0, 0.0, 0.0)), ("H", (0.0, 0.0, 1.2))]) mol.run_pyscf() @@ -145,66 +258,3 @@ def electronic_energy(parameters): lih_uccsd_energy = -7.847535097575567 assert vqe.fun == pytest.approx(lih_uccsd_energy) - - -def test_ucc_bk(): - """Build a UCC-BK circuit""" - # Control - gate_list = [ - gates.H(0), - gates.RX(1, -0.5 * np.pi, trainable=False), - gates.H(2), - gates.CNOT(2, 1), - gates.CNOT(1, 0), - gates.RZ(0, 0.0), - gates.CNOT(1, 0), - gates.CNOT(2, 1), - gates.H(0), - gates.RX(1, -0.5 * np.pi, trainable=False), - gates.H(2), - gates.RX(0, -0.5 * np.pi, trainable=False), - gates.RX(1, -0.5 * np.pi, trainable=False), - gates.RX(2, -0.5 * np.pi, trainable=False), - gates.CNOT(2, 1), - gates.CNOT(1, 0), - gates.RZ(0, 0.0), - gates.CNOT(1, 0), - gates.CNOT(2, 1), - gates.RX(0, -0.5 * np.pi, trainable=False), - gates.RX(1, -0.5 * np.pi, trainable=False), - gates.RX(2, -0.5 * np.pi, trainable=False), - ] - # Test ucc_function - circuit = ucc_circuit(4, [0, 2], ferm_qubit_map="bk") - # Check gates are correct - assert all( - control.name == test.name and control.target_qubits == test.target_qubits - for control, test in zip(gate_list, list(circuit.queue)) - ) - # Check that only two parametrised gates - assert len(circuit.get_parameters()) == 2 - - -def test_ucc_ferm_qubit_map_error(): - """If unknown fermion to qubit map used""" - with pytest.raises(KeyError): - ucc_circuit(2, [0, 1], ferm_qubit_map="Unknown") - - -# def test_ucc_ansatz_default_args(): -# mol = Molecule([("H", (0.0, 0.0, 0.0)), ("H", (0.0, 0.0, 0.7))]) -# mol.run_pyscf() -# -# # Build control circuit -# control_circuit = hf_circuit(4, 2) -# for excitation in ([0, 1, 2, 3], [0, 2], [1, 3]): -# control_circuit += ucc_circuit(4, excitation) -# -# test_circuit = ucc_ansatz(mol) -# -# assert all( -# control.name == test.name and control.target_qubits == test.target_qubits -# for control, test in zip(list(control_circuit.queue), list(test_circuit.queue)) -# ) -# # Check that number of parametrised gates is the same -# assert len(control_circuit.get_parameters()) == len(test_circuit.get_parameters()) From 95cbc78c5ce439b46516c374cd748cf0d55fd8a9 Mon Sep 17 00:00:00 2001 From: Wong Zi Cheng <70616433+chmwzc@users.noreply.github.com> Date: Wed, 3 Jan 2024 06:32:29 +0000 Subject: [PATCH 06/11] Remove VQE-UCC test to shorten overall test times --- tests/test_ucc.py | 62 ----------------------------------------------- 1 file changed, 62 deletions(-) diff --git a/tests/test_ucc.py b/tests/test_ucc.py index 1c759e1..c767e7d 100644 --- a/tests/test_ucc.py +++ b/tests/test_ucc.py @@ -196,65 +196,3 @@ def test_ucc_ansatz_h2(): ) # Check that number of parametrised gates is the same assert len(control_circuit.get_parameters()) == len(test_circuit.get_parameters()) - - -def test_ucc_ansatz_h2(): - # Define molecule and populate - mol = Molecule([("Li", (0.0, 0.0, 0.0)), ("H", (0.0, 0.0, 1.2))]) - mol.run_pyscf() - - # Apply embedding - active_space = [1, 5] - mol.hf_embedding(active=[1, 5]) - hamiltonian = mol.hamiltonian(oei=mol.embed_oei, tei=mol.embed_tei, constant=mol.inactive_energy) - - # Set parameters for the rest of the experiment - n_qubits = mol.n_active_orbs - n_electrons = mol.n_active_e - - excitations = generate_excitations(2, tuple(range(n_electrons)), tuple(range(n_electrons, n_qubits))) - excitations += generate_excitations(1, tuple(range(n_electrons)), tuple(range(n_electrons, n_qubits))) - n_excitations = len(excitations) - - # Build circuit - circuit = hf_circuit(n_qubits, n_electrons) - - # UCCSD: Circuit - all_coeffs = [] - for _ex in excitations: - coeffs = [] - circuit += ucc_circuit(n_qubits, _ex, coeffs=coeffs) - all_coeffs.append(coeffs) - - def electronic_energy(parameters): - r""" - Loss function for the UCCSD ansatz - """ - all_parameters = [] - - # UCC parameters - # Expand the parameters to match the total UCC ansatz manually - _ucc = parameters[:n_excitations] - # Manually group the related excitations together - ucc_parameters = [_ucc[0], _ucc[1], _ucc[2]] - # Need to iterate through each excitation this time - for _coeffs, _parameter in zip(all_coeffs, ucc_parameters): - # Convert a single value to a array with dimension=n_param_gates - ucc_parameter = np.repeat(_parameter, len(_coeffs)) - # Multiply by coeffs - ucc_parameter *= _coeffs - all_parameters.append(ucc_parameter) - - # Flatten all_parameters into a single list to set the circuit parameters - all_parameters = [_x for _param in all_parameters for _x in _param] - circuit.set_parameters(all_parameters) - - return expectation(circuit, hamiltonian) - - # Random initialization - params = np.zeros(n_excitations) - vqe = minimize(electronic_energy, params) - - lih_uccsd_energy = -7.847535097575567 - - assert vqe.fun == pytest.approx(lih_uccsd_energy) From ae95e7d711e167ff3eb1db45cf3a82e1c62cd931 Mon Sep 17 00:00:00 2001 From: Wong Zi Cheng <70616433+chmwzc@users.noreply.github.com> Date: Wed, 3 Jan 2024 07:18:12 +0000 Subject: [PATCH 07/11] Add tests for ucc_ansatz function --- tests/test_ucc.py | 92 ++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 91 insertions(+), 1 deletion(-) diff --git a/tests/test_ucc.py b/tests/test_ucc.py index c767e7d..acf8f06 100644 --- a/tests/test_ucc.py +++ b/tests/test_ucc.py @@ -6,7 +6,7 @@ from scipy.optimize import minimize from qibochem.ansatz.hf_reference import hf_circuit -from qibochem.ansatz.ucc import ( # ucc_ansatz +from qibochem.ansatz.ucc import ( generate_excitations, mp2_amplitude, sort_excitations, @@ -178,6 +178,23 @@ def test_ucc_ferm_qubit_map_error(): ucc_circuit(2, [0, 1], ferm_qubit_map="Unknown") +def test_ucc_parameter_coefficients(): + """Coefficients used to multiply the parameters in the UCC circuit. Note: may change in future!""" + # UCC-JW singles + control_values = (-1.0, 1.0) + coeffs = [] + circuit = ucc_circuit(2, [0, 1], coeffs=coeffs) + # Check that the signs of the coefficients have been saved + assert all(control == test for control, test in zip(control_values, coeffs)) + + # UCC-JW doubles + control_values = (-0.25, 0.25, 0.25, 0.25, -0.25, -0.25, -0.25, 0.25) + coeffs = [] + circuit = ucc_circuit(4, [0, 1, 2, 3], coeffs=coeffs) + # Check that the signs of the coefficients have been saved + assert all(control == test for control, test in zip(control_values, coeffs)) + + def test_ucc_ansatz_h2(): """Test the default arguments of ucc_ansatz using H2""" mol = Molecule([("H", (0.0, 0.0, 0.0)), ("H", (0.0, 0.0, 0.7))]) @@ -196,3 +213,76 @@ def test_ucc_ansatz_h2(): ) # Check that number of parametrised gates is the same assert len(control_circuit.get_parameters()) == len(test_circuit.get_parameters()) + + +def test_ucc_ansatz_embedding(): + """Test the default arguments of ucc_ansatz using LiH with HF embedding applied, but without the HF circuit""" + mol = Molecule([("Li", (0.0, 0.0, 0.0)), ("H", (0.0, 0.0, 1.4))]) + mol.run_pyscf() + mol.hf_embedding(active=[1, 2, 5]) + + # Generate all possible excitations + excitations = [] + for order in range(2, 0, -1): + # 2 electrons, 6 spin-orbitals + excitations += sort_excitations(generate_excitations(order, range(0, 2), range(2, 6))) + # Build control circuit + control_circuit = hf_circuit(6, 0) + for excitation in excitations: + control_circuit += ucc_circuit(6, excitation) + + test_circuit = ucc_ansatz(mol, include_hf=False) + + assert all( + control.name == test.name and control.target_qubits == test.target_qubits + for control, test in zip(list(control_circuit.queue), list(test_circuit.queue)) + ) + # Check that number of parametrised gates is the same + assert len(control_circuit.get_parameters()) == len(test_circuit.get_parameters()) + + +def test_ucc_ansatz_excitations(): + """Test the `excitations` argument of ucc_ansatz""" + mol = Molecule([("Li", (0.0, 0.0, 0.0)), ("H", (0.0, 0.0, 1.4))]) + mol.run_pyscf() + mol.hf_embedding(active=[1, 2, 5]) + + # Generate all possible excitations + excitations = [[0, 1, 2, 3], [0, 1, 4, 5]] + # Build control circuit + control_circuit = hf_circuit(6, 2) + for excitation in excitations: + control_circuit += ucc_circuit(6, excitation) + + test_circuit = ucc_ansatz(mol, excitations=excitations) + + assert all( + control.name == test.name and control.target_qubits == test.target_qubits + for control, test in zip(list(control_circuit.queue), list(test_circuit.queue)) + ) + # Check that number of parametrised gates is the same + assert len(control_circuit.get_parameters()) == len(test_circuit.get_parameters()) + + +def test_ucc_ansatz_error_checks(): + """Test the checks for input validity""" + mol = Molecule([("Li", (0.0, 0.0, 0.0)), ("H", (0.0, 0.0, 1.4))]) + # Define number of electrons and spin-obritals by hand + mol.nelec = (2, 2) + mol.nso = 12 + + # Excitation level check: Input is in ("S", "D", "T", "Q") + with pytest.raises(AssertionError): + ucc_ansatz(mol, "Z") + + # Excitation level check: Excitation > doubles + with pytest.raises(NotImplementedError): + ucc_ansatz(mol, "T") + + # Excitations list check: excitation must have an even number of elements + with pytest.raises(AssertionError): + ucc_ansatz(mol, excitations=[[0]]) + + # Input parameter check: Must have correct number of input parameters + with pytest.raises(AssertionError): + ucc_ansatz(mol, excitations=[[0, 1, 2, 3]], thetas=np.zeros(2)) From 0407d4e58e7bde300a730516a5c3f2cf9ef11538 Mon Sep 17 00:00:00 2001 From: Wong Zi Cheng <70616433+chmwzc@users.noreply.github.com> Date: Wed, 3 Jan 2024 08:06:31 +0000 Subject: [PATCH 08/11] Use MP2 amplitudes to initialse ucc_ansatz --- src/qibochem/ansatz/ucc.py | 8 ++++++-- tests/test_ucc.py | 17 ++++++++++++++++- 2 files changed, 22 insertions(+), 3 deletions(-) diff --git a/src/qibochem/ansatz/ucc.py b/src/qibochem/ansatz/ucc.py index aca17d4..63f8d15 100644 --- a/src/qibochem/ansatz/ucc.py +++ b/src/qibochem/ansatz/ucc.py @@ -276,6 +276,7 @@ def ucc_ansatz( trotter_steps=1, ferm_qubit_map=None, include_hf=True, + use_mp2_guess=True, ): """ Build a circuit corresponding to the UCC ansatz with multiple excitations for a given Molecule. @@ -293,6 +294,7 @@ def ucc_ansatz( theta=theta/trotter_steps. Default is 1 ferm_qubit_map: fermion-to-qubit transformation. Default: Jordan-Wigner ("jw") include_hf: Whether or not to start the circuit with a Hartree-Fock circuit. Default: ``True`` + use_mp2_guesses: Whether or not to use MP2 amplitudes as the initial guess parameter. Default: ``True`` Returns: circuit: Qibo Circuit corresponding to the UCC ansatz @@ -326,12 +328,14 @@ def ucc_ansatz( assert all(len(_ex) % 2 == 0 for _ex in excitations), "Excitation with an odd number of elements found!" # Check if thetas argument given, define to be all zeros if not - # TODO: Unsure if want to use MP2 guess amplitudes for the doubles? Some say good, some say bad # Number of circuit parameters: S->2, D->8, (T/Q->32/128; Not sure?) n_parameters = 2 * len([_ex for _ex in excitations if len(_ex) == 2]) # Singles n_parameters += 8 * len([_ex for _ex in excitations if len(_ex) == 4]) # Doubles if thetas is None: - thetas = np.zeros(n_parameters) + if use_mp2_guess: + thetas = np.array([mp2_amplitude(excitation, molecule.eps, molecule.tei) for excitation in excitations]) + else: + thetas = np.zeros(n_parameters) else: # Check that number of circuit variables (i.e. thetas) matches the number of circuit parameters assert len(thetas) == n_parameters, "Number of input parameters doesn't match the number of circuit parameters!" diff --git a/tests/test_ucc.py b/tests/test_ucc.py index acf8f06..9765ef7 100644 --- a/tests/test_ucc.py +++ b/tests/test_ucc.py @@ -202,7 +202,8 @@ def test_ucc_ansatz_h2(): # Build control circuit control_circuit = hf_circuit(4, 2) - for excitation in ([0, 1, 2, 3], [0, 2], [1, 3]): + excitations = ([0, 1, 2, 3], [0, 2], [1, 3]) + for excitation in excitations: control_circuit += ucc_circuit(4, excitation) test_circuit = ucc_ansatz(mol) @@ -214,6 +215,20 @@ def test_ucc_ansatz_h2(): # Check that number of parametrised gates is the same assert len(control_circuit.get_parameters()) == len(test_circuit.get_parameters()) + # Then check that the circuit parameters are the MP2 guess parameters + # Get the MP2 amplitudes first, then expand the list based on the excitation type + mp2_guess_amplitudes = [] + for excitation in excitations: + mp2_guess_amplitudes += [ + mp2_amplitude(excitation, mol.eps, mol.tei) for _ in range(2 ** (2 * (len(excitation) // 2) - 1)) + ] + mp2_guess_amplitudes = np.array(mp2_guess_amplitudes) + coeffs = np.array([-0.25, 0.25, 0.25, 0.25, -0.25, -0.25, -0.25, 0.25, 0.0, 0.0, 0.0, 0.0]) + mp2_guess_amplitudes *= coeffs + # Need to flatten the output of circuit.get_parameters() to compare it to mp2_guess_amplitudes + test_parameters = np.array([_x for _tuple in test_circuit.get_parameters() for _x in _tuple]) + assert np.allclose(mp2_guess_amplitudes, test_parameters) + def test_ucc_ansatz_embedding(): """Test the default arguments of ucc_ansatz using LiH with HF embedding applied, but without the HF circuit""" From 9c5003e39d34136b14ac81a62ae1a1dd972c620d Mon Sep 17 00:00:00 2001 From: Wong Zi Cheng <70616433+chmwzc@users.noreply.github.com> Date: Wed, 3 Jan 2024 08:43:06 +0000 Subject: [PATCH 09/11] Documentation for ucc_ansatz Also added remaining test coverage --- doc/source/api-reference/ansatz.rst | 1 + src/qibochem/ansatz/basis_rotation.py | 2 +- src/qibochem/ansatz/hardware_efficient.py | 2 +- src/qibochem/ansatz/hf_reference.py | 2 +- src/qibochem/ansatz/ucc.py | 30 +++++++++++------------ tests/test_ucc.py | 6 ++++- 6 files changed, 24 insertions(+), 19 deletions(-) diff --git a/doc/source/api-reference/ansatz.rst b/doc/source/api-reference/ansatz.rst index 48ecacd..fb931f8 100644 --- a/doc/source/api-reference/ansatz.rst +++ b/doc/source/api-reference/ansatz.rst @@ -21,6 +21,7 @@ Unitary Coupled Cluster .. autofunction:: qibochem.ansatz.ucc.ucc_circuit +.. autofunction:: qibochem.ansatz.ucc.ucc_ansatz Basis rotation -------------- diff --git a/src/qibochem/ansatz/basis_rotation.py b/src/qibochem/ansatz/basis_rotation.py index 887480f..c5a540a 100644 --- a/src/qibochem/ansatz/basis_rotation.py +++ b/src/qibochem/ansatz/basis_rotation.py @@ -129,7 +129,7 @@ def br_circuit(n_qubits, parameters, n_occ): n_occ: Number of occupied orbitals Returns: - Qibo ``Circuit`` corresponding to the basis rotation ansatz between the occupied and virtual orbitals + Qibo ``Circuit``: Circuit corresponding to the basis rotation ansatz between the occupied and virtual orbitals """ assert len(parameters) == (n_occ * (n_qubits - n_occ)), "Need len(parameters) == (n_occ * n_virt)" # Unitary rotation matrix \exp(\kappa) diff --git a/src/qibochem/ansatz/hardware_efficient.py b/src/qibochem/ansatz/hardware_efficient.py index 5b13e17..89a5fef 100644 --- a/src/qibochem/ansatz/hardware_efficient.py +++ b/src/qibochem/ansatz/hardware_efficient.py @@ -15,7 +15,7 @@ def hea(n_layers, n_qubits, parameter_gates=["RY", "RZ"], coupling_gates="CZ"): valid two-qubit ``Qibo`` gate. Default: ``"CZ"`` Returns: - List of gates corresponding to the hardware-efficient ansatz + ``list``: Gates corresponding to the hardware-efficient ansatz """ gate_list = [] diff --git a/src/qibochem/ansatz/hf_reference.py b/src/qibochem/ansatz/hf_reference.py index 5683a50..db64f6a 100644 --- a/src/qibochem/ansatz/hf_reference.py +++ b/src/qibochem/ansatz/hf_reference.py @@ -59,7 +59,7 @@ def hf_circuit(n_qubits, n_electrons, ferm_qubit_map=None): ferm_qubit_map: Fermion to qubit map. Must be either Jordan-Wigner (``jw``) or Brayvi-Kitaev (``bk``). Default value is ``jw``. Returns: - Qibo ``Circuit`` initialized in a HF reference state + Qibo ``Circuit``: Circuit initialized in a HF reference state """ # Which fermion-to-qubit map to use if ferm_qubit_map is None: diff --git a/src/qibochem/ansatz/ucc.py b/src/qibochem/ansatz/ucc.py index 63f8d15..a234c1b 100644 --- a/src/qibochem/ansatz/ucc.py +++ b/src/qibochem/ansatz/ucc.py @@ -68,13 +68,13 @@ def ucc_circuit(n_qubits, excitation, theta=0.0, trotter_steps=1, ferm_qubit_map excitation: Iterable of orbitals involved in the excitation; must have an even number of elements E.g. ``[0, 1, 2, 3]`` represents the excitation of electrons in orbitals ``(0, 1)`` to ``(2, 3)`` theta: UCC parameter. Defaults to 0.0 - trotter_steps: Number of Trotter steps; i.e. number of times the UCC ansatz is applied with \theta = \theta / trotter_steps + trotter_steps: Number of Trotter steps; i.e. number of times the UCC ansatz is applied with ``theta`` = ``theta / trotter_steps``. Default: 1 ferm_qubit_map: Fermion-to-qubit transformation. Default is Jordan-Wigner (``jw``). coeffs: List to hold the coefficients for the rotation parameter in each Pauli string. May be useful in running the VQE. WARNING: Will be modified in this function Returns: - Qibo ``Circuit`` corresponding to a single UCC excitation + Qibo ``Circuit``: Circuit corresponding to a single UCC excitation """ # Check size of orbitals input n_orbitals = len(excitation) @@ -279,25 +279,25 @@ def ucc_ansatz( use_mp2_guess=True, ): """ - Build a circuit corresponding to the UCC ansatz with multiple excitations for a given Molecule. - If no excitations are given, it defaults to returning the full UCCSD circuit ansatz for the - Molecule. + Convenience function for buildng a circuit corresponding to the UCC ansatz with multiple excitations for a given ``Molecule``. + If no excitations are given, it defaults to returning the full UCCSD circuit ansatz. Args: - molecule: Molecule of interest. - excitation_level: Include excitations up to how many electrons, i.e. ("S", "D", "T", "Q") - Ignored if `excitations` argument is given. Default is "D", i.e. double excitations - excitations: List of excitations (e.g. `[[0, 1, 2, 3], [0, 1, 4, 5]]`) used to build the - UCC circuit. Overrides the `excitation_level` argument - thetas: Parameters for the excitations. Defaults to an array of zeros if not given + molecule: The ``Molecule`` of interest. + excitation_level: Include excitations up to how many electrons, i.e. ``"S"`` or ``"D"``. + Ignored if ``excitations`` argument is given. Default: ``"D"``, i.e. double excitations + excitations: List of excitations (e.g. ``[[0, 1, 2, 3], [0, 1, 4, 5]]``) used to build the + UCC circuit. Overrides the ``excitation_level`` argument + thetas: Parameters for the excitations. Default value depends on the ``use_mp2_guess`` argument. trotter_steps: number of Trotter steps; i.e. number of times the UCC ansatz is applied with - theta=theta/trotter_steps. Default is 1 - ferm_qubit_map: fermion-to-qubit transformation. Default: Jordan-Wigner ("jw") + ``theta`` = ``theta / trotter_steps``. Default: 1 + ferm_qubit_map: fermion-to-qubit transformation. Default: Jordan-Wigner (``"jw"``) include_hf: Whether or not to start the circuit with a Hartree-Fock circuit. Default: ``True`` - use_mp2_guesses: Whether or not to use MP2 amplitudes as the initial guess parameter. Default: ``True`` + use_mp2_guess: Whether to use MP2 amplitudes or a numpy zero array as the initial guess parameter. Default: ``True``; + use the MP2 amplitudes as the default guess parameters Returns: - circuit: Qibo Circuit corresponding to the UCC ansatz + Qibo ``Circuit``: Circuit corresponding to an UCC ansatz """ # Get the number of electrons and spin-orbitals from the molecule argument n_elec = sum(molecule.nelec) if molecule.n_active_e is None else molecule.n_active_e diff --git a/tests/test_ucc.py b/tests/test_ucc.py index 9765ef7..9a5d688 100644 --- a/tests/test_ucc.py +++ b/tests/test_ucc.py @@ -246,7 +246,7 @@ def test_ucc_ansatz_embedding(): for excitation in excitations: control_circuit += ucc_circuit(6, excitation) - test_circuit = ucc_ansatz(mol, include_hf=False) + test_circuit = ucc_ansatz(mol, include_hf=False, use_mp2_guess=False) assert all( control.name == test.name and control.target_qubits == test.target_qubits @@ -255,6 +255,10 @@ def test_ucc_ansatz_embedding(): # Check that number of parametrised gates is the same assert len(control_circuit.get_parameters()) == len(test_circuit.get_parameters()) + # Check that the circuit parameters are all zeros + test_parameters = np.array([_x for _tuple in test_circuit.get_parameters() for _x in _tuple]) + assert np.allclose(test_parameters, np.zeros(len(test_parameters))) + def test_ucc_ansatz_excitations(): """Test the `excitations` argument of ucc_ansatz""" From d8a043cf07d0dec48b67a32592afdcd464ac8af4 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 22 Jan 2024 02:11:54 +0000 Subject: [PATCH 10/11] Bump jinja2 from 3.1.2 to 3.1.3 Bumps [jinja2](https://github.com/pallets/jinja) from 3.1.2 to 3.1.3. - [Release notes](https://github.com/pallets/jinja/releases) - [Changelog](https://github.com/pallets/jinja/blob/main/CHANGES.rst) - [Commits](https://github.com/pallets/jinja/compare/3.1.2...3.1.3) --- updated-dependencies: - dependency-name: jinja2 dependency-type: indirect ... Signed-off-by: dependabot[bot] --- poetry.lock | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/poetry.lock b/poetry.lock index 190e391..0389c59 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,4 +1,4 @@ -# This file is automatically @generated by Poetry 1.6.1 and should not be changed by hand. +# This file is automatically @generated by Poetry 1.7.1 and should not be changed by hand. [[package]] name = "alabaster" @@ -1103,13 +1103,13 @@ testing = ["Django", "attrs", "colorama", "docopt", "pytest (<7.0.0)"] [[package]] name = "jinja2" -version = "3.1.2" +version = "3.1.3" description = "A very fast and expressive template engine." optional = false python-versions = ">=3.7" files = [ - {file = "Jinja2-3.1.2-py3-none-any.whl", hash = "sha256:6088930bfe239f0e6710546ab9c19c9ef35e29792895fed6e6e31a023a182a61"}, - {file = "Jinja2-3.1.2.tar.gz", hash = "sha256:31351a702a408a9e7595a8fc6150fc3f43bb6bf7e319770cbc0db9df9437e852"}, + {file = "Jinja2-3.1.3-py3-none-any.whl", hash = "sha256:7d6d50dd97d52cbc355597bd845fabfbac3f551e1f99619e39a35ce8c370b5fa"}, + {file = "Jinja2-3.1.3.tar.gz", hash = "sha256:ac8bd6544d4bb2c9792bf3a159e80bba8fda7f07e81bc3aed565432d5925ba90"}, ] [package.dependencies] @@ -2359,6 +2359,7 @@ files = [ {file = "PyYAML-6.0.1-cp311-cp311-win_amd64.whl", hash = "sha256:bf07ee2fef7014951eeb99f56f39c9bb4af143d8aa3c21b1677805985307da34"}, {file = "PyYAML-6.0.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:855fb52b0dc35af121542a76b9a84f8d1cd886ea97c84703eaa6d88e37a2ad28"}, {file = "PyYAML-6.0.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:40df9b996c2b73138957fe23a16a4f0ba614f4c0efce1e9406a184b6d07fa3a9"}, + {file = "PyYAML-6.0.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a08c6f0fe150303c1c6b71ebcd7213c2858041a7e01975da3a99aed1e7a378ef"}, {file = "PyYAML-6.0.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6c22bec3fbe2524cde73d7ada88f6566758a8f7227bfbf93a408a9d86bcc12a0"}, {file = "PyYAML-6.0.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8d4e9c88387b0f5c7d5f281e55304de64cf7f9c0021a3525bd3b1c542da3b0e4"}, {file = "PyYAML-6.0.1-cp312-cp312-win32.whl", hash = "sha256:d483d2cdf104e7c9fa60c544d92981f12ad66a457afae824d146093b8c294c54"}, From dff78dbab7105414a879687aa48b23cfe1b406f6 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Tue, 23 Jan 2024 05:43:02 +0000 Subject: [PATCH 11/11] Bump pillow from 10.1.0 to 10.2.0 Bumps [pillow](https://github.com/python-pillow/Pillow) from 10.1.0 to 10.2.0. - [Release notes](https://github.com/python-pillow/Pillow/releases) - [Changelog](https://github.com/python-pillow/Pillow/blob/main/CHANGES.rst) - [Commits](https://github.com/python-pillow/Pillow/compare/10.1.0...10.2.0) --- updated-dependencies: - dependency-name: pillow dependency-type: indirect ... Signed-off-by: dependabot[bot] --- poetry.lock | 128 ++++++++++++++++++++++++++++++---------------------- 1 file changed, 73 insertions(+), 55 deletions(-) diff --git a/poetry.lock b/poetry.lock index 0389c59..5146096 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1840,70 +1840,88 @@ ptyprocess = ">=0.5" [[package]] name = "pillow" -version = "10.1.0" +version = "10.2.0" description = "Python Imaging Library (Fork)" optional = false python-versions = ">=3.8" files = [ - {file = "Pillow-10.1.0-cp310-cp310-macosx_10_10_x86_64.whl", hash = "sha256:1ab05f3db77e98f93964697c8efc49c7954b08dd61cff526b7f2531a22410106"}, - {file = "Pillow-10.1.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:6932a7652464746fcb484f7fc3618e6503d2066d853f68a4bd97193a3996e273"}, - {file = "Pillow-10.1.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a5f63b5a68daedc54c7c3464508d8c12075e56dcfbd42f8c1bf40169061ae666"}, - {file = "Pillow-10.1.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c0949b55eb607898e28eaccb525ab104b2d86542a85c74baf3a6dc24002edec2"}, - {file = "Pillow-10.1.0-cp310-cp310-manylinux_2_28_aarch64.whl", hash = "sha256:ae88931f93214777c7a3aa0a8f92a683f83ecde27f65a45f95f22d289a69e593"}, - {file = "Pillow-10.1.0-cp310-cp310-manylinux_2_28_x86_64.whl", hash = "sha256:b0eb01ca85b2361b09480784a7931fc648ed8b7836f01fb9241141b968feb1db"}, - {file = "Pillow-10.1.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:d27b5997bdd2eb9fb199982bb7eb6164db0426904020dc38c10203187ae2ff2f"}, - {file = "Pillow-10.1.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:7df5608bc38bd37ef585ae9c38c9cd46d7c81498f086915b0f97255ea60c2818"}, - {file = "Pillow-10.1.0-cp310-cp310-win_amd64.whl", hash = "sha256:41f67248d92a5e0a2076d3517d8d4b1e41a97e2df10eb8f93106c89107f38b57"}, - {file = "Pillow-10.1.0-cp311-cp311-macosx_10_10_x86_64.whl", hash = "sha256:1fb29c07478e6c06a46b867e43b0bcdb241b44cc52be9bc25ce5944eed4648e7"}, - {file = "Pillow-10.1.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:2cdc65a46e74514ce742c2013cd4a2d12e8553e3a2563c64879f7c7e4d28bce7"}, - {file = "Pillow-10.1.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:50d08cd0a2ecd2a8657bd3d82c71efd5a58edb04d9308185d66c3a5a5bed9610"}, - {file = "Pillow-10.1.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:062a1610e3bc258bff2328ec43f34244fcec972ee0717200cb1425214fe5b839"}, - {file = "Pillow-10.1.0-cp311-cp311-manylinux_2_28_aarch64.whl", hash = "sha256:61f1a9d247317fa08a308daaa8ee7b3f760ab1809ca2da14ecc88ae4257d6172"}, - {file = "Pillow-10.1.0-cp311-cp311-manylinux_2_28_x86_64.whl", hash = "sha256:a646e48de237d860c36e0db37ecaecaa3619e6f3e9d5319e527ccbc8151df061"}, - {file = "Pillow-10.1.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:47e5bf85b80abc03be7455c95b6d6e4896a62f6541c1f2ce77a7d2bb832af262"}, - {file = "Pillow-10.1.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:a92386125e9ee90381c3369f57a2a50fa9e6aa8b1cf1d9c4b200d41a7dd8e992"}, - {file = "Pillow-10.1.0-cp311-cp311-win_amd64.whl", hash = "sha256:0f7c276c05a9767e877a0b4c5050c8bee6a6d960d7f0c11ebda6b99746068c2a"}, - {file = "Pillow-10.1.0-cp312-cp312-macosx_10_10_x86_64.whl", hash = "sha256:a89b8312d51715b510a4fe9fc13686283f376cfd5abca8cd1c65e4c76e21081b"}, - {file = "Pillow-10.1.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:00f438bb841382b15d7deb9a05cc946ee0f2c352653c7aa659e75e592f6fa17d"}, - {file = "Pillow-10.1.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3d929a19f5469b3f4df33a3df2983db070ebb2088a1e145e18facbc28cae5b27"}, - {file = "Pillow-10.1.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9a92109192b360634a4489c0c756364c0c3a2992906752165ecb50544c251312"}, - {file = "Pillow-10.1.0-cp312-cp312-manylinux_2_28_aarch64.whl", hash = "sha256:0248f86b3ea061e67817c47ecbe82c23f9dd5d5226200eb9090b3873d3ca32de"}, - {file = "Pillow-10.1.0-cp312-cp312-manylinux_2_28_x86_64.whl", hash = "sha256:9882a7451c680c12f232a422730f986a1fcd808da0fd428f08b671237237d651"}, - {file = "Pillow-10.1.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:1c3ac5423c8c1da5928aa12c6e258921956757d976405e9467c5f39d1d577a4b"}, - {file = "Pillow-10.1.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:806abdd8249ba3953c33742506fe414880bad78ac25cc9a9b1c6ae97bedd573f"}, - {file = "Pillow-10.1.0-cp312-cp312-win_amd64.whl", hash = "sha256:eaed6977fa73408b7b8a24e8b14e59e1668cfc0f4c40193ea7ced8e210adf996"}, - {file = "Pillow-10.1.0-cp38-cp38-macosx_10_10_x86_64.whl", hash = "sha256:fe1e26e1ffc38be097f0ba1d0d07fcade2bcfd1d023cda5b29935ae8052bd793"}, - {file = "Pillow-10.1.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:7a7e3daa202beb61821c06d2517428e8e7c1aab08943e92ec9e5755c2fc9ba5e"}, - {file = "Pillow-10.1.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:24fadc71218ad2b8ffe437b54876c9382b4a29e030a05a9879f615091f42ffc2"}, - {file = "Pillow-10.1.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fa1d323703cfdac2036af05191b969b910d8f115cf53093125e4058f62012c9a"}, - {file = "Pillow-10.1.0-cp38-cp38-manylinux_2_28_aarch64.whl", hash = "sha256:912e3812a1dbbc834da2b32299b124b5ddcb664ed354916fd1ed6f193f0e2d01"}, - {file = "Pillow-10.1.0-cp38-cp38-manylinux_2_28_x86_64.whl", hash = "sha256:7dbaa3c7de82ef37e7708521be41db5565004258ca76945ad74a8e998c30af8d"}, - {file = "Pillow-10.1.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:9d7bc666bd8c5a4225e7ac71f2f9d12466ec555e89092728ea0f5c0c2422ea80"}, - {file = "Pillow-10.1.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:baada14941c83079bf84c037e2d8b7506ce201e92e3d2fa0d1303507a8538212"}, - {file = "Pillow-10.1.0-cp38-cp38-win_amd64.whl", hash = "sha256:2ef6721c97894a7aa77723740a09547197533146fba8355e86d6d9a4a1056b14"}, - {file = "Pillow-10.1.0-cp39-cp39-macosx_10_10_x86_64.whl", hash = "sha256:0a026c188be3b443916179f5d04548092e253beb0c3e2ee0a4e2cdad72f66099"}, - {file = "Pillow-10.1.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:04f6f6149f266a100374ca3cc368b67fb27c4af9f1cc8cb6306d849dcdf12616"}, - {file = "Pillow-10.1.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bb40c011447712d2e19cc261c82655f75f32cb724788df315ed992a4d65696bb"}, - {file = "Pillow-10.1.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1a8413794b4ad9719346cd9306118450b7b00d9a15846451549314a58ac42219"}, - {file = "Pillow-10.1.0-cp39-cp39-manylinux_2_28_aarch64.whl", hash = "sha256:c9aeea7b63edb7884b031a35305629a7593272b54f429a9869a4f63a1bf04c34"}, - {file = "Pillow-10.1.0-cp39-cp39-manylinux_2_28_x86_64.whl", hash = "sha256:b4005fee46ed9be0b8fb42be0c20e79411533d1fd58edabebc0dd24626882cfd"}, - {file = "Pillow-10.1.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:4d0152565c6aa6ebbfb1e5d8624140a440f2b99bf7afaafbdbf6430426497f28"}, - {file = "Pillow-10.1.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:d921bc90b1defa55c9917ca6b6b71430e4286fc9e44c55ead78ca1a9f9eba5f2"}, - {file = "Pillow-10.1.0-cp39-cp39-win_amd64.whl", hash = "sha256:cfe96560c6ce2f4c07d6647af2d0f3c54cc33289894ebd88cfbb3bcd5391e256"}, - {file = "Pillow-10.1.0-pp310-pypy310_pp73-macosx_10_10_x86_64.whl", hash = "sha256:937bdc5a7f5343d1c97dc98149a0be7eb9704e937fe3dc7140e229ae4fc572a7"}, - {file = "Pillow-10.1.0-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b1c25762197144e211efb5f4e8ad656f36c8d214d390585d1d21281f46d556ba"}, - {file = "Pillow-10.1.0-pp310-pypy310_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:afc8eef765d948543a4775f00b7b8c079b3321d6b675dde0d02afa2ee23000b4"}, - {file = "Pillow-10.1.0-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:883f216eac8712b83a63f41b76ddfb7b2afab1b74abbb413c5df6680f071a6b9"}, - {file = "Pillow-10.1.0-pp39-pypy39_pp73-macosx_10_10_x86_64.whl", hash = "sha256:b920e4d028f6442bea9a75b7491c063f0b9a3972520731ed26c83e254302eb1e"}, - {file = "Pillow-10.1.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1c41d960babf951e01a49c9746f92c5a7e0d939d1652d7ba30f6b3090f27e412"}, - {file = "Pillow-10.1.0-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:1fafabe50a6977ac70dfe829b2d5735fd54e190ab55259ec8aea4aaea412fa0b"}, - {file = "Pillow-10.1.0-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:3b834f4b16173e5b92ab6566f0473bfb09f939ba14b23b8da1f54fa63e4b623f"}, - {file = "Pillow-10.1.0.tar.gz", hash = "sha256:e6bf8de6c36ed96c86ea3b6e1d5273c53f46ef518a062464cd7ef5dd2cf92e38"}, + {file = "pillow-10.2.0-cp310-cp310-macosx_10_10_x86_64.whl", hash = "sha256:7823bdd049099efa16e4246bdf15e5a13dbb18a51b68fa06d6c1d4d8b99a796e"}, + {file = "pillow-10.2.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:83b2021f2ade7d1ed556bc50a399127d7fb245e725aa0113ebd05cfe88aaf588"}, + {file = "pillow-10.2.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6fad5ff2f13d69b7e74ce5b4ecd12cc0ec530fcee76356cac6742785ff71c452"}, + {file = "pillow-10.2.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:da2b52b37dad6d9ec64e653637a096905b258d2fc2b984c41ae7d08b938a67e4"}, + {file = "pillow-10.2.0-cp310-cp310-manylinux_2_28_aarch64.whl", hash = "sha256:47c0995fc4e7f79b5cfcab1fc437ff2890b770440f7696a3ba065ee0fd496563"}, + {file = "pillow-10.2.0-cp310-cp310-manylinux_2_28_x86_64.whl", hash = "sha256:322bdf3c9b556e9ffb18f93462e5f749d3444ce081290352c6070d014c93feb2"}, + {file = "pillow-10.2.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:51f1a1bffc50e2e9492e87d8e09a17c5eea8409cda8d3f277eb6edc82813c17c"}, + {file = "pillow-10.2.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:69ffdd6120a4737710a9eee73e1d2e37db89b620f702754b8f6e62594471dee0"}, + {file = "pillow-10.2.0-cp310-cp310-win32.whl", hash = "sha256:c6dafac9e0f2b3c78df97e79af707cdc5ef8e88208d686a4847bab8266870023"}, + {file = "pillow-10.2.0-cp310-cp310-win_amd64.whl", hash = "sha256:aebb6044806f2e16ecc07b2a2637ee1ef67a11840a66752751714a0d924adf72"}, + {file = "pillow-10.2.0-cp310-cp310-win_arm64.whl", hash = "sha256:7049e301399273a0136ff39b84c3678e314f2158f50f517bc50285fb5ec847ad"}, + {file = "pillow-10.2.0-cp311-cp311-macosx_10_10_x86_64.whl", hash = "sha256:35bb52c37f256f662abdfa49d2dfa6ce5d93281d323a9af377a120e89a9eafb5"}, + {file = "pillow-10.2.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:9c23f307202661071d94b5e384e1e1dc7dfb972a28a2310e4ee16103e66ddb67"}, + {file = "pillow-10.2.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:773efe0603db30c281521a7c0214cad7836c03b8ccff897beae9b47c0b657d61"}, + {file = "pillow-10.2.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:11fa2e5984b949b0dd6d7a94d967743d87c577ff0b83392f17cb3990d0d2fd6e"}, + {file = "pillow-10.2.0-cp311-cp311-manylinux_2_28_aarch64.whl", hash = "sha256:716d30ed977be8b37d3ef185fecb9e5a1d62d110dfbdcd1e2a122ab46fddb03f"}, + {file = "pillow-10.2.0-cp311-cp311-manylinux_2_28_x86_64.whl", hash = "sha256:a086c2af425c5f62a65e12fbf385f7c9fcb8f107d0849dba5839461a129cf311"}, + {file = "pillow-10.2.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:c8de2789052ed501dd829e9cae8d3dcce7acb4777ea4a479c14521c942d395b1"}, + {file = "pillow-10.2.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:609448742444d9290fd687940ac0b57fb35e6fd92bdb65386e08e99af60bf757"}, + {file = "pillow-10.2.0-cp311-cp311-win32.whl", hash = "sha256:823ef7a27cf86df6597fa0671066c1b596f69eba53efa3d1e1cb8b30f3533068"}, + {file = "pillow-10.2.0-cp311-cp311-win_amd64.whl", hash = "sha256:1da3b2703afd040cf65ec97efea81cfba59cdbed9c11d8efc5ab09df9509fc56"}, + {file = "pillow-10.2.0-cp311-cp311-win_arm64.whl", hash = "sha256:edca80cbfb2b68d7b56930b84a0e45ae1694aeba0541f798e908a49d66b837f1"}, + {file = "pillow-10.2.0-cp312-cp312-macosx_10_10_x86_64.whl", hash = "sha256:1b5e1b74d1bd1b78bc3477528919414874748dd363e6272efd5abf7654e68bef"}, + {file = "pillow-10.2.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:0eae2073305f451d8ecacb5474997c08569fb4eb4ac231ffa4ad7d342fdc25ac"}, + {file = "pillow-10.2.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b7c2286c23cd350b80d2fc9d424fc797575fb16f854b831d16fd47ceec078f2c"}, + {file = "pillow-10.2.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1e23412b5c41e58cec602f1135c57dfcf15482013ce6e5f093a86db69646a5aa"}, + {file = "pillow-10.2.0-cp312-cp312-manylinux_2_28_aarch64.whl", hash = "sha256:52a50aa3fb3acb9cf7213573ef55d31d6eca37f5709c69e6858fe3bc04a5c2a2"}, + {file = "pillow-10.2.0-cp312-cp312-manylinux_2_28_x86_64.whl", hash = "sha256:127cee571038f252a552760076407f9cff79761c3d436a12af6000cd182a9d04"}, + {file = "pillow-10.2.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:8d12251f02d69d8310b046e82572ed486685c38f02176bd08baf216746eb947f"}, + {file = "pillow-10.2.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:54f1852cd531aa981bc0965b7d609f5f6cc8ce8c41b1139f6ed6b3c54ab82bfb"}, + {file = "pillow-10.2.0-cp312-cp312-win32.whl", hash = "sha256:257d8788df5ca62c980314053197f4d46eefedf4e6175bc9412f14412ec4ea2f"}, + {file = "pillow-10.2.0-cp312-cp312-win_amd64.whl", hash = "sha256:154e939c5f0053a383de4fd3d3da48d9427a7e985f58af8e94d0b3c9fcfcf4f9"}, + {file = "pillow-10.2.0-cp312-cp312-win_arm64.whl", hash = "sha256:f379abd2f1e3dddb2b61bc67977a6b5a0a3f7485538bcc6f39ec76163891ee48"}, + {file = "pillow-10.2.0-cp38-cp38-macosx_10_10_x86_64.whl", hash = "sha256:8373c6c251f7ef8bda6675dd6d2b3a0fcc31edf1201266b5cf608b62a37407f9"}, + {file = "pillow-10.2.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:870ea1ada0899fd0b79643990809323b389d4d1d46c192f97342eeb6ee0b8483"}, + {file = "pillow-10.2.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b4b6b1e20608493548b1f32bce8cca185bf0480983890403d3b8753e44077129"}, + {file = "pillow-10.2.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3031709084b6e7852d00479fd1d310b07d0ba82765f973b543c8af5061cf990e"}, + {file = "pillow-10.2.0-cp38-cp38-manylinux_2_28_aarch64.whl", hash = "sha256:3ff074fc97dd4e80543a3e91f69d58889baf2002b6be64347ea8cf5533188213"}, + {file = "pillow-10.2.0-cp38-cp38-manylinux_2_28_x86_64.whl", hash = "sha256:cb4c38abeef13c61d6916f264d4845fab99d7b711be96c326b84df9e3e0ff62d"}, + {file = "pillow-10.2.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:b1b3020d90c2d8e1dae29cf3ce54f8094f7938460fb5ce8bc5c01450b01fbaf6"}, + {file = "pillow-10.2.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:170aeb00224ab3dc54230c797f8404507240dd868cf52066f66a41b33169bdbe"}, + {file = "pillow-10.2.0-cp38-cp38-win32.whl", hash = "sha256:c4225f5220f46b2fde568c74fca27ae9771536c2e29d7c04f4fb62c83275ac4e"}, + {file = "pillow-10.2.0-cp38-cp38-win_amd64.whl", hash = "sha256:0689b5a8c5288bc0504d9fcee48f61a6a586b9b98514d7d29b840143d6734f39"}, + {file = "pillow-10.2.0-cp39-cp39-macosx_10_10_x86_64.whl", hash = "sha256:b792a349405fbc0163190fde0dc7b3fef3c9268292586cf5645598b48e63dc67"}, + {file = "pillow-10.2.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:c570f24be1e468e3f0ce7ef56a89a60f0e05b30a3669a459e419c6eac2c35364"}, + {file = "pillow-10.2.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d8ecd059fdaf60c1963c58ceb8997b32e9dc1b911f5da5307aab614f1ce5c2fb"}, + {file = "pillow-10.2.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c365fd1703040de1ec284b176d6af5abe21b427cb3a5ff68e0759e1e313a5e7e"}, + {file = "pillow-10.2.0-cp39-cp39-manylinux_2_28_aarch64.whl", hash = "sha256:70c61d4c475835a19b3a5aa42492409878bbca7438554a1f89d20d58a7c75c01"}, + {file = "pillow-10.2.0-cp39-cp39-manylinux_2_28_x86_64.whl", hash = "sha256:b6f491cdf80ae540738859d9766783e3b3c8e5bd37f5dfa0b76abdecc5081f13"}, + {file = "pillow-10.2.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:9d189550615b4948f45252d7f005e53c2040cea1af5b60d6f79491a6e147eef7"}, + {file = "pillow-10.2.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:49d9ba1ed0ef3e061088cd1e7538a0759aab559e2e0a80a36f9fd9d8c0c21591"}, + {file = "pillow-10.2.0-cp39-cp39-win32.whl", hash = "sha256:babf5acfede515f176833ed6028754cbcd0d206f7f614ea3447d67c33be12516"}, + {file = "pillow-10.2.0-cp39-cp39-win_amd64.whl", hash = "sha256:0304004f8067386b477d20a518b50f3fa658a28d44e4116970abfcd94fac34a8"}, + {file = "pillow-10.2.0-cp39-cp39-win_arm64.whl", hash = "sha256:0fb3e7fc88a14eacd303e90481ad983fd5b69c761e9e6ef94c983f91025da869"}, + {file = "pillow-10.2.0-pp310-pypy310_pp73-macosx_10_10_x86_64.whl", hash = "sha256:322209c642aabdd6207517e9739c704dc9f9db943015535783239022002f054a"}, + {file = "pillow-10.2.0-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3eedd52442c0a5ff4f887fab0c1c0bb164d8635b32c894bc1faf4c618dd89df2"}, + {file = "pillow-10.2.0-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:cb28c753fd5eb3dd859b4ee95de66cc62af91bcff5db5f2571d32a520baf1f04"}, + {file = "pillow-10.2.0-pp310-pypy310_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:33870dc4653c5017bf4c8873e5488d8f8d5f8935e2f1fb9a2208c47cdd66efd2"}, + {file = "pillow-10.2.0-pp310-pypy310_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:3c31822339516fb3c82d03f30e22b1d038da87ef27b6a78c9549888f8ceda39a"}, + {file = "pillow-10.2.0-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:a2b56ba36e05f973d450582fb015594aaa78834fefe8dfb8fcd79b93e64ba4c6"}, + {file = "pillow-10.2.0-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:d8e6aeb9201e655354b3ad049cb77d19813ad4ece0df1249d3c793de3774f8c7"}, + {file = "pillow-10.2.0-pp39-pypy39_pp73-macosx_10_10_x86_64.whl", hash = "sha256:2247178effb34a77c11c0e8ac355c7a741ceca0a732b27bf11e747bbc950722f"}, + {file = "pillow-10.2.0-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:15587643b9e5eb26c48e49a7b33659790d28f190fc514a322d55da2fb5c2950e"}, + {file = "pillow-10.2.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:753cd8f2086b2b80180d9b3010dd4ed147efc167c90d3bf593fe2af21265e5a5"}, + {file = "pillow-10.2.0-pp39-pypy39_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:7c8f97e8e7a9009bcacbe3766a36175056c12f9a44e6e6f2d5caad06dcfbf03b"}, + {file = "pillow-10.2.0-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:d1b35bcd6c5543b9cb547dee3150c93008f8dd0f1fef78fc0cd2b141c5baf58a"}, + {file = "pillow-10.2.0-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:fe4c15f6c9285dc54ce6553a3ce908ed37c8f3825b5a51a15c91442bb955b868"}, + {file = "pillow-10.2.0.tar.gz", hash = "sha256:e87f0b2c78157e12d7686b27d63c070fd65d994e8ddae6f328e0dcf4a0cd007e"}, ] [package.extras] docs = ["furo", "olefile", "sphinx (>=2.4)", "sphinx-copybutton", "sphinx-inline-tabs", "sphinx-removed-in", "sphinxext-opengraph"] +fpx = ["olefile"] +mic = ["olefile"] tests = ["check-manifest", "coverage", "defusedxml", "markdown2", "olefile", "packaging", "pyroma", "pytest", "pytest-cov", "pytest-timeout"] +typing = ["typing-extensions"] +xmp = ["defusedxml"] [[package]] name = "platformdirs"