diff --git a/Cargo.lock b/Cargo.lock index fc7b5c9d3..6f5090090 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -101,9 +101,9 @@ dependencies = [ [[package]] name = "anyhow" -version = "1.0.94" +version = "1.0.95" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c1fd03a028ef38ba2276dce7e33fcd6369c158a1bca17946c4b1b701891c1ff7" +checksum = "34ac096ce696dc2fcabef30516bb13c0a68a11d30131d3df6f04711467681b04" [[package]] name = "arbitrary" @@ -636,9 +636,9 @@ dependencies = [ [[package]] name = "clap" -version = "4.5.22" +version = "4.5.23" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "69371e34337c4c984bbe322360c2547210bf632eb2814bbe78a6e87a2935bd2b" +checksum = "3135e7ec2ef7b10c6ed8950f0f792ed96ee093fa088608f1c76e569722700c84" dependencies = [ "clap_builder", "clap_derive", @@ -646,9 +646,9 @@ dependencies = [ [[package]] name = "clap_builder" -version = "4.5.22" +version = "4.5.23" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6e24c1b4099818523236a8ca881d2b45db98dadfb4625cf6608c12069fcbbde1" +checksum = "30582fc632330df2bd26877bde0c1f4470d57c582bbc070376afcd04d8cb4838" dependencies = [ "anstream", "anstyle", @@ -670,9 +670,9 @@ dependencies = [ [[package]] name = "clap_lex" -version = "0.7.2" +version = "0.7.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1462739cb27611015575c0c11df5df7601141071f07518d56fcc1be504cbec97" +checksum = "f46ad14479a25103f283c0f10005961cf086d8dc42205bb44c46ac563475dca6" [[package]] name = "clvm-derive" @@ -2026,9 +2026,9 @@ dependencies = [ [[package]] name = "pyo3" -version = "0.22.6" +version = "0.23.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f402062616ab18202ae8319da13fa4279883a2b8a9d9f83f20dbade813ce1884" +checksum = "e484fd2c8b4cb67ab05a318f1fd6fa8f199fcc30819f08f07d200809dba26c15" dependencies = [ "cfg-if", "indoc", @@ -2046,9 +2046,9 @@ dependencies = [ [[package]] name = "pyo3-build-config" -version = "0.22.6" +version = "0.23.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b14b5775b5ff446dd1056212d778012cbe8a0fbffd368029fd9e25b514479c38" +checksum = "dc0e0469a84f208e20044b98965e1561028180219e35352a2afaf2b942beff3b" dependencies = [ "once_cell", "target-lexicon", @@ -2056,9 +2056,9 @@ dependencies = [ [[package]] name = "pyo3-ffi" -version = "0.22.6" +version = "0.23.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9ab5bcf04a2cdcbb50c7d6105de943f543f9ed92af55818fd17b660390fc8636" +checksum = "eb1547a7f9966f6f1a0f0227564a9945fe36b90da5a93b3933fc3dc03fae372d" dependencies = [ "libc", "pyo3-build-config", @@ -2066,9 +2066,9 @@ dependencies = [ [[package]] name = "pyo3-macros" -version = "0.22.6" +version = "0.23.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0fd24d897903a9e6d80b968368a34e1525aeb719d568dba8b3d4bfa5dc67d453" +checksum = "fdb6da8ec6fa5cedd1626c886fc8749bdcbb09424a86461eb8cdf096b7c33257" dependencies = [ "proc-macro2", "pyo3-macros-backend", @@ -2078,9 +2078,9 @@ dependencies = [ [[package]] name = "pyo3-macros-backend" -version = "0.22.6" +version = "0.23.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "36c011a03ba1e50152b4b394b479826cad97e7a21eb52df179cd91ac411cbfbe" +checksum = "38a385202ff5a92791168b1136afae5059d3ac118457bb7bc304c197c2d33e7d" dependencies = [ "heck", "proc-macro2", @@ -2150,9 +2150,9 @@ dependencies = [ [[package]] name = "rcgen" -version = "0.13.1" +version = "0.13.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "54077e1872c46788540de1ea3d7f4ccb1983d12f9aa909b234468676c1a36779" +checksum = "75e669e5202259b5314d1ea5397316ad400819437857b90861765f24c4cf80a2" dependencies = [ "pem", "ring", @@ -2486,9 +2486,9 @@ checksum = "13c2bddecc57b384dee18652358fb23172facb8a2c51ccc10d74c157bdea3292" [[package]] name = "syn" -version = "2.0.90" +version = "2.0.95" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "919d3b74a5dd0ccd15aeb8f93e7006bd9e14c295087c9896a110f490752bcf31" +checksum = "46f71c0377baf4ef1cc3e3402ded576dccc315800fbc62dfc7fe04b009773b4a" dependencies = [ "proc-macro2", "quote", diff --git a/Cargo.toml b/Cargo.toml index 73332b09b..e62c5d246 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -129,16 +129,16 @@ clvm-traits-fuzz = { path = "./crates/clvm-traits/fuzz", version = "0.16.0" } clvm-utils-fuzz = { path = "./crates/clvm-utils/fuzz", version = "0.16.0" } blst = { version = "0.3.12", features = ["portable"] } clvmr = "0.10.0" -syn = "2.0.90" +syn = "2.0.95" quote = "1.0.32" proc-macro2 = "1.0.92" proc-macro-crate = "1.3.1" -anyhow = "1.0.94" +anyhow = "1.0.95" sha2 = "0.10.8" hkdf = "0.12.0" hex = "0.4.3" thiserror = "1.0.69" -pyo3 = "0.22.6" +pyo3 = "0.23.3" arbitrary = "1.4.1" rand = "0.8.5" criterion = "0.5.1" @@ -153,11 +153,11 @@ num-traits = "0.2.15" num-bigint = "0.4.5" text-diff = "0.4.0" lazy_static = "1.4.0" -rcgen = "0.13.1" +rcgen = "0.13.2" rsa = "0.9.7" time = "0.3.22" rusqlite = "0.31.0" -clap = "4.5.22" +clap = "4.5.23" zstd = "0.13.2" blocking-threadpool = "1.0.1" libfuzzer-sys = "0.4" diff --git a/LICENSE b/LICENSE index 7e8bb6bb2..ef0cf9fb3 100644 --- a/LICENSE +++ b/LICENSE @@ -186,7 +186,7 @@ same "printed page" as the copyright notice for easier identification within third-party archives. - Copyright 2024 Chia Network Inc. + Copyright 2025 Chia Network Inc. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. diff --git a/crates/chia-bls/fuzz/fuzz_targets/blspy-fidelity.rs b/crates/chia-bls/fuzz/fuzz_targets/blspy-fidelity.rs index 37beb1a60..3d4f385c4 100644 --- a/crates/chia-bls/fuzz/fuzz_targets/blspy-fidelity.rs +++ b/crates/chia-bls/fuzz/fuzz_targets/blspy-fidelity.rs @@ -35,7 +35,7 @@ fuzz_target!(|data: &[u8]| { print(sys.executable) "#, None, None).unwrap(); */ - let blspy = py.import_bound("blspy").unwrap(); + let blspy = py.import("blspy").unwrap(); let aug = blspy.getattr("AugSchemeMPL").unwrap(); // Generate key pair from seed @@ -43,7 +43,7 @@ fuzz_target!(|data: &[u8]| { let py_sk = aug .call_method1( "key_gen", - PyTuple::new_bound(py, [PyBytes::new_bound(py, data)]), + PyTuple::new(py, [PyBytes::new(py, data)]).unwrap(), ) .unwrap(); @@ -66,10 +66,14 @@ fuzz_target!(|data: &[u8]| { let py_sk1 = aug .call_method1( "derive_child_sk_unhardened", - PyTuple::new_bound( + PyTuple::new( py, - [py_sk.clone(), idx.to_object(py).bind(py).clone().into_any()], - ), + [ + py_sk.clone(), + idx.into_pyobject(py).unwrap().clone().into_any(), + ], + ) + .unwrap(), ) .unwrap(); assert_eq!(to_bytes(&py_sk1), rust_sk1.to_bytes()); @@ -78,7 +82,11 @@ fuzz_target!(|data: &[u8]| { let py_pk1 = aug .call_method1( "derive_child_pk_unhardened", - PyTuple::new_bound(py, [py_pk, idx.to_object(py).bind(py).clone().into_any()]), + PyTuple::new( + py, + [py_pk, idx.into_pyobject(py).unwrap().clone().into_any()], + ) + .unwrap(), ) .unwrap(); assert_eq!(to_bytes(&py_pk1), rust_pk1.to_bytes()); @@ -88,7 +96,7 @@ fuzz_target!(|data: &[u8]| { let py_sig1 = aug .call_method1( "sign", - PyTuple::new_bound(py, [py_sk1, PyBytes::new_bound(py, data).into_any()]), + PyTuple::new(py, [py_sk1, PyBytes::new(py, data).into_any()]).unwrap(), ) .unwrap(); assert_eq!(to_bytes(&py_sig1), rust_sig1.to_bytes()); @@ -99,7 +107,11 @@ fuzz_target!(|data: &[u8]| { let py_sk2 = aug .call_method1( "derive_child_sk", - PyTuple::new_bound(py, [py_sk, idx.to_object(py).bind(py).clone().into_any()]), + PyTuple::new( + py, + [py_sk, idx.into_pyobject(py).unwrap().clone().into_any()], + ) + .unwrap(), ) .unwrap(); assert_eq!(to_bytes(&py_sk2), rust_sk2.to_bytes()); @@ -109,7 +121,7 @@ fuzz_target!(|data: &[u8]| { let py_sig2 = aug .call_method1( "sign", - PyTuple::new_bound(py, [py_sk2, PyBytes::new_bound(py, data).into_any()]), + PyTuple::new(py, [py_sk2, PyBytes::new(py, data).into_any()]).unwrap(), ) .unwrap(); assert_eq!(to_bytes(&py_sig2), rust_sig2.to_bytes()); @@ -119,7 +131,7 @@ fuzz_target!(|data: &[u8]| { let py_agg = aug .call_method1( "aggregate", - PyTuple::new_bound(py, [PyList::new_bound(py, [py_sig1, py_sig2])]), + PyTuple::new(py, [PyList::new(py, [py_sig1, py_sig2]).unwrap()]).unwrap(), ) .unwrap(); assert_eq!(to_bytes(&py_agg), rust_agg.to_bytes()); diff --git a/crates/chia-bls/src/bls_cache.rs b/crates/chia-bls/src/bls_cache.rs index 188acc183..101256b98 100644 --- a/crates/chia-bls/src/bls_cache.rs +++ b/crates/chia-bls/src/bls_cache.rs @@ -163,12 +163,12 @@ impl BlsCache { sig: &Signature, ) -> PyResult { let pks = pks - .iter()? + .try_iter()? .map(|item| item?.extract()) .collect::>>()?; let msgs = msgs - .iter()? + .try_iter()? .map(|item| item?.extract()) .collect::>>()?; @@ -184,10 +184,13 @@ impl BlsCache { pub fn py_items(&self, py: pyo3::Python<'_>) -> PyResult { use pyo3::prelude::*; use pyo3::types::PyBytes; - let ret = PyList::empty_bound(py); + let ret = PyList::empty(py); let c = self.cache.lock().expect("cache"); for (key, value) in &c.items { - ret.append((PyBytes::new_bound(py, key), value.clone().into_py(py)))?; + ret.append(( + PyBytes::new(py, key), + value.clone().into_pyobject(py)?.into_any(), + ))?; } Ok(ret.into()) } @@ -195,7 +198,7 @@ impl BlsCache { #[pyo3(name = "update")] pub fn py_update(&self, other: &Bound<'_, PySequence>) -> PyResult<()> { let mut c = self.cache.lock().expect("cache"); - for item in other.borrow().iter()? { + for item in other.borrow().try_iter()? { let (key, value): (Vec, GTElement) = item?.extract()?; c.put( key.try_into() @@ -209,11 +212,11 @@ impl BlsCache { #[pyo3(name = "evict")] pub fn py_evict(&self, pks: &Bound<'_, PyList>, msgs: &Bound<'_, PyList>) -> PyResult<()> { let pks = pks - .iter()? + .try_iter()? .map(|item| item?.extract()) .collect::>>()?; let msgs = msgs - .iter()? + .try_iter()? .map(|item| item?.extract()) .collect::>>()?; self.evict(pks.into_iter().zip(msgs)); diff --git a/crates/chia-bls/src/gtelement.rs b/crates/chia-bls/src/gtelement.rs index 373890c2f..30dad4d63 100644 --- a/crates/chia-bls/src/gtelement.rs +++ b/crates/chia-bls/src/gtelement.rs @@ -148,7 +148,10 @@ mod pybindings { impl ToJsonDict for GTElement { fn to_json_dict(&self, py: Python<'_>) -> PyResult { let bytes = self.to_bytes(); - Ok(("0x".to_string() + &hex::encode(bytes)).into_py(py)) + Ok(("0x".to_string() + &hex::encode(bytes)) + .into_pyobject(py)? + .into_any() + .unbind()) } } diff --git a/crates/chia-bls/src/public_key.rs b/crates/chia-bls/src/public_key.rs index 8c6140083..8d1eed6e9 100644 --- a/crates/chia-bls/src/public_key.rs +++ b/crates/chia-bls/src/public_key.rs @@ -371,7 +371,10 @@ mod pybindings { impl ToJsonDict for PublicKey { fn to_json_dict(&self, py: Python<'_>) -> PyResult { let bytes = self.to_bytes(); - Ok(("0x".to_string() + &hex::encode(bytes)).into_py(py)) + Ok(("0x".to_string() + &hex::encode(bytes)) + .into_pyobject(py)? + .into_any() + .unbind()) } } @@ -734,7 +737,7 @@ mod tests { mod pytests { use super::*; use crate::SecretKey; - use pyo3::{IntoPy, Python}; + use pyo3::Python; use rand::rngs::StdRng; use rand::{Rng, SeedableRng}; use rstest::rstest; @@ -750,7 +753,7 @@ mod pytests { let pk = sk.public_key(); Python::with_gil(|py| { let string = pk.to_json_dict(py).expect("to_json_dict"); - let py_class = py.get_type_bound::(); + let py_class = py.get_type::(); let pk2: PublicKey = PublicKey::from_json_dict(&py_class, py, string.bind(py)) .unwrap() .extract(py) @@ -769,11 +772,14 @@ mod pytests { fn test_json_dict(#[case] input: &str, #[case] msg: &str) { pyo3::prepare_freethreaded_python(); Python::with_gil(|py| { - let py_class = py.get_type_bound::(); - let err = - PublicKey::from_json_dict(&py_class, py, input.to_string().into_py(py).bind(py)) - .unwrap_err(); - assert_eq!(err.value_bound(py).to_string(), msg.to_string()); + let py_class = py.get_type::(); + let err = PublicKey::from_json_dict( + &py_class, + py, + &input.to_string().into_pyobject(py).unwrap().into_any(), + ) + .unwrap_err(); + assert_eq!(err.value(py).to_string(), msg.to_string()); }); } } diff --git a/crates/chia-bls/src/secret_key.rs b/crates/chia-bls/src/secret_key.rs index 9c9998122..935af33a9 100644 --- a/crates/chia-bls/src/secret_key.rs +++ b/crates/chia-bls/src/secret_key.rs @@ -311,7 +311,10 @@ mod pybindings { impl ToJsonDict for SecretKey { fn to_json_dict(&self, py: Python<'_>) -> PyResult { let bytes = self.to_bytes(); - Ok(("0x".to_string() + &hex::encode(bytes)).into_py(py)) + Ok(("0x".to_string() + &hex::encode(bytes)) + .into_pyobject(py)? + .into_any() + .unbind()) } } @@ -556,7 +559,7 @@ mod tests { #[cfg(feature = "py-bindings")] mod pytests { use super::*; - use pyo3::{IntoPy, Python}; + use pyo3::Python; use rand::rngs::StdRng; use rand::{Rng, SeedableRng}; use rstest::rstest; @@ -571,7 +574,7 @@ mod pytests { let sk = SecretKey::from_seed(&data); Python::with_gil(|py| { let string = sk.to_json_dict(py).expect("to_json_dict"); - let py_class = py.get_type_bound::(); + let py_class = py.get_type::(); let sk2 = SecretKey::from_json_dict(&py_class, py, string.bind(py)) .unwrap() .extract(py) @@ -606,11 +609,14 @@ mod pytests { fn test_json_dict(#[case] input: &str, #[case] msg: &str) { pyo3::prepare_freethreaded_python(); Python::with_gil(|py| { - let py_class = py.get_type_bound::(); - let err = - SecretKey::from_json_dict(&py_class, py, input.to_string().into_py(py).bind(py)) - .unwrap_err(); - assert_eq!(err.value_bound(py).to_string(), msg.to_string()); + let py_class = py.get_type::(); + let err = SecretKey::from_json_dict( + &py_class, + py, + &input.to_string().into_pyobject(py).unwrap().into_any(), + ) + .unwrap_err(); + assert_eq!(err.value(py).to_string(), msg.to_string()); }); } } diff --git a/crates/chia-bls/src/signature.rs b/crates/chia-bls/src/signature.rs index 01eed3b9b..5000c1b3a 100644 --- a/crates/chia-bls/src/signature.rs +++ b/crates/chia-bls/src/signature.rs @@ -536,7 +536,10 @@ mod pybindings { impl ToJsonDict for Signature { fn to_json_dict(&self, py: Python<'_>) -> PyResult { let bytes = self.to_bytes(); - Ok(("0x".to_string() + &hex::encode(bytes)).into_py(py)) + Ok(("0x".to_string() + &hex::encode(bytes)) + .into_pyobject(py)? + .into_any() + .unbind()) } } @@ -1254,7 +1257,7 @@ mod tests { mod pytests { use super::*; - use pyo3::{IntoPy, Python}; + use pyo3::Python; use rand::rngs::StdRng; use rand::{Rng, SeedableRng}; use rstest::rstest; @@ -1272,7 +1275,7 @@ mod pytests { let sig = sign(&sk, msg); Python::with_gil(|py| { let string = sig.to_json_dict(py).expect("to_json_dict"); - let py_class = py.get_type_bound::(); + let py_class = py.get_type::(); let sig2 = Signature::from_json_dict(&py_class, py, string.bind(py)) .unwrap() .extract(py) @@ -1291,11 +1294,14 @@ mod pytests { fn test_json_dict(#[case] input: &str, #[case] msg: &str) { pyo3::prepare_freethreaded_python(); Python::with_gil(|py| { - let py_class = py.get_type_bound::(); - let err = - Signature::from_json_dict(&py_class, py, input.to_string().into_py(py).bind(py)) - .unwrap_err(); - assert_eq!(err.value_bound(py).to_string(), msg.to_string()); + let py_class = py.get_type::(); + let err = Signature::from_json_dict( + &py_class, + py, + &input.to_string().into_pyobject(py).unwrap().into_any(), + ) + .unwrap_err(); + assert_eq!(err.value(py).to_string(), msg.to_string()); }); } } diff --git a/crates/chia-consensus/src/merkle_tree.rs b/crates/chia-consensus/src/merkle_tree.rs index 52a3d53b3..e031d5c8a 100644 --- a/crates/chia-consensus/src/merkle_tree.rs +++ b/crates/chia-consensus/src/merkle_tree.rs @@ -368,7 +368,7 @@ impl MerkleSet { included_leaf: [u8; 32], ) -> PyResult<(bool, PyObject)> { match self.generate_proof(&included_leaf) { - Ok((included, proof)) => Ok((included, PyBytes::new_bound(py, &proof).into())), + Ok((included, proof)) => Ok((included, PyBytes::new(py, &proof).into())), Err(_) => Err(PyValueError::new_err("invalid proof")), } } diff --git a/crates/chia-datalayer/src/merkle.rs b/crates/chia-datalayer/src/merkle.rs index 03d6a53c1..4d5e1f90d 100644 --- a/crates/chia-datalayer/src/merkle.rs +++ b/crates/chia-datalayer/src/merkle.rs @@ -1,7 +1,7 @@ #[cfg(feature = "py-bindings")] use pyo3::{ - buffer::PyBuffer, exceptions::PyValueError, pyclass, pymethods, FromPyObject, IntoPy, PyObject, - PyResult, Python, + buffer::PyBuffer, exceptions::PyValueError, pyclass, pymethods, types::PyInt, Bound, + FromPyObject, IntoPyObject, PyAny, PyErr, PyResult, Python, }; use chia_protocol::Bytes32; @@ -20,9 +20,13 @@ use thiserror::Error; pub struct TreeIndex(u32); #[cfg(feature = "py-bindings")] -impl IntoPy for TreeIndex { - fn into_py(self, py: Python<'_>) -> PyObject { - self.0.into_py(py) +impl<'py> IntoPyObject<'py> for TreeIndex { + type Target = PyInt; + type Output = Bound<'py, Self::Target>; + type Error = std::convert::Infallible; + + fn into_pyobject(self, py: Python<'py>) -> Result { + self.0.into_pyobject(py) } } @@ -42,9 +46,13 @@ type Hash = Bytes32; pub struct KvId(i64); #[cfg(feature = "py-bindings")] -impl IntoPy for KvId { - fn into_py(self, py: Python<'_>) -> PyObject { - self.0.into_py(py) +impl<'py> IntoPyObject<'py> for KvId { + type Target = PyInt; + type Output = Bound<'py, Self::Target>; + type Error = std::convert::Infallible; + + fn into_pyobject(self, py: Python<'py>) -> Result { + self.0.into_pyobject(py) } } @@ -314,11 +322,15 @@ impl Node { } #[cfg(feature = "py-bindings")] -impl IntoPy for Node { - fn into_py(self, py: Python<'_>) -> PyObject { +impl<'py> IntoPyObject<'py> for Node { + type Target = PyAny; + type Output = Bound<'py, Self::Target>; + type Error = PyErr; + + fn into_pyobject(self, py: Python<'py>) -> Result { match self { - Node::Internal(node) => node.into_py(py), - Node::Leaf(node) => node.into_py(py), + Node::Internal(node) => Ok(node.into_pyobject(py)?.into_any()), + Node::Leaf(node) => Ok(node.into_pyobject(py)?.into_any()), } } } @@ -1279,15 +1291,14 @@ impl MerkleBlob { index: TreeIndex, py: Python<'_>, ) -> PyResult { - let list = pyo3::types::PyList::empty_bound(py); + let list = pyo3::types::PyList::empty(py); for (index, node) in self .get_lineage_with_indexes(index) .map_err(|e| PyValueError::new_err(e.to_string()))? { - use pyo3::conversion::IntoPy; use pyo3::types::PyListMethods; - list.append((index.into_py(py), node.into_py(py)))?; + list.append((index.into_pyobject(py)?, node.into_pyobject(py)?))?; } Ok(list.into()) @@ -1295,13 +1306,12 @@ impl MerkleBlob { #[pyo3(name = "get_nodes_with_indexes")] pub fn py_get_nodes_with_indexes(&self, py: Python<'_>) -> PyResult { - let list = pyo3::types::PyList::empty_bound(py); + let list = pyo3::types::PyList::empty(py); for item in MerkleBlobParentFirstIterator::new(&self.blob) { - use pyo3::conversion::IntoPy; use pyo3::types::PyListMethods; let (index, block) = item.map_err(|e| PyValueError::new_err(e.to_string()))?; - list.append((index.into_py(py), block.node.into_py(py)))?; + list.append((index.into_pyobject(py)?, block.node.into_pyobject(py)?))?; } Ok(list.into()) diff --git a/crates/chia-protocol/src/block_record.rs b/crates/chia-protocol/src/block_record.rs index 2478fbe21..c90e0d77b 100644 --- a/crates/chia-protocol/src/block_record.rs +++ b/crates/chia-protocol/src/block_record.rs @@ -132,12 +132,12 @@ impl BlockRecord { } fn sp_iters_impl(&self, py: Python<'_>, constants: &Bound<'_, PyAny>) -> PyResult { - let ctx = PyDict::new_bound(py); + let ctx = PyDict::new(py); ctx.set_item("sub_slot_iters", self.sub_slot_iters)?; ctx.set_item("signage_point_index", self.signage_point_index)?; ctx.set_item("constants", constants)?; - py.run_bound( - "from chia.consensus.pot_iterations import calculate_ip_iters, calculate_sp_iters\n\ + py.run( + c"from chia.consensus.pot_iterations import calculate_ip_iters, calculate_sp_iters\n\ ret = calculate_sp_iters(constants, sub_slot_iters, signage_point_index)\n", None, Some(&ctx), @@ -146,13 +146,13 @@ impl BlockRecord { } fn ip_iters_impl(&self, py: Python<'_>, constants: &Bound<'_, PyAny>) -> PyResult { - let ctx = PyDict::new_bound(py); + let ctx = PyDict::new(py); ctx.set_item("sub_slot_iters", self.sub_slot_iters)?; ctx.set_item("signage_point_index", self.signage_point_index)?; ctx.set_item("required_iters", self.required_iters)?; ctx.set_item("constants", constants)?; - py.run_bound( - "from chia.consensus.pot_iterations import calculate_ip_iters, calculate_sp_iters\n\ + py.run( + c"from chia.consensus.pot_iterations import calculate_ip_iters, calculate_sp_iters\n\ ret = calculate_ip_iters(constants, sub_slot_iters, signage_point_index, required_iters)\n", None, Some(&ctx), diff --git a/crates/chia-protocol/src/bytes.rs b/crates/chia-protocol/src/bytes.rs index 74f5c7962..74f567829 100644 --- a/crates/chia-protocol/src/bytes.rs +++ b/crates/chia-protocol/src/bytes.rs @@ -86,7 +86,7 @@ impl Streamable for Bytes { #[cfg(feature = "py-bindings")] impl ToJsonDict for Bytes { fn to_json_dict(&self, py: Python<'_>) -> PyResult { - Ok(format!("0x{self}").to_object(py)) + Ok(format!("0x{self}").into_pyobject(py)?.into_any().unbind()) } } @@ -226,7 +226,7 @@ impl Streamable for BytesImpl { #[cfg(feature = "py-bindings")] impl ToJsonDict for BytesImpl { fn to_json_dict(&self, py: Python<'_>) -> PyResult { - Ok(format!("0x{self}").to_object(py)) + Ok(format!("0x{self}").into_pyobject(py)?.into_any().unbind()) } } @@ -390,16 +390,13 @@ impl From for Bytes32 { } #[cfg(feature = "py-bindings")] -impl ToPyObject for BytesImpl { - fn to_object(&self, py: Python<'_>) -> PyObject { - ChiaToPython::to_python(self, py).unwrap().into() - } -} +impl<'py, const N: usize> IntoPyObject<'py> for BytesImpl { + type Target = PyAny; + type Output = Bound<'py, Self::Target>; + type Error = PyErr; -#[cfg(feature = "py-bindings")] -impl IntoPy for BytesImpl { - fn into_py(self, py: Python<'_>) -> PyObject { - ChiaToPython::to_python(&self, py).unwrap().into() + fn into_pyobject(self, py: Python<'py>) -> Result { + ChiaToPython::to_python(&self, py) } } @@ -407,15 +404,15 @@ impl IntoPy for BytesImpl { impl ChiaToPython for BytesImpl { fn to_python<'a>(&self, py: Python<'a>) -> PyResult> { if N == 32 { - let bytes_module = PyModule::import_bound(py, "chia_rs.sized_bytes")?; + let bytes_module = PyModule::import(py, "chia_rs.sized_bytes")?; let ty = bytes_module.getattr("bytes32")?; - ty.call1((self.0.into_py(py),)) + ty.call1((self.0.into_pyobject(py)?,)) } else if N == 48 { - let bytes_module = PyModule::import_bound(py, "chia_rs.sized_bytes")?; + let bytes_module = PyModule::import(py, "chia_rs.sized_bytes")?; let ty = bytes_module.getattr("bytes48")?; - ty.call1((self.0.into_py(py),)) + ty.call1((self.0.into_pyobject(py)?,)) } else { - Ok(PyBytes::new_bound(py, &self.0).into_any()) + Ok(PyBytes::new(py, &self.0).into_any()) } } } @@ -431,23 +428,20 @@ impl<'py, const N: usize> FromPyObject<'py> for BytesImpl { } #[cfg(feature = "py-bindings")] -impl ToPyObject for Bytes { - fn to_object(&self, py: Python<'_>) -> PyObject { - PyBytes::new_bound(py, &self.0).into() - } -} +impl<'py> IntoPyObject<'py> for Bytes { + type Target = PyAny; + type Output = Bound<'py, Self::Target>; + type Error = std::convert::Infallible; -#[cfg(feature = "py-bindings")] -impl IntoPy for Bytes { - fn into_py(self, py: Python<'_>) -> PyObject { - PyBytes::new_bound(py, &self.0).into() + fn into_pyobject(self, py: Python<'py>) -> Result { + Ok(PyBytes::new(py, &self.0).into_any()) } } #[cfg(feature = "py-bindings")] impl ChiaToPython for Bytes { fn to_python<'a>(&self, py: Python<'a>) -> PyResult> { - Ok(PyBytes::new_bound(py, &self.0).into_any()) + Ok(PyBytes::new(py, &self.0).into_any()) } } diff --git a/crates/chia-protocol/src/chia_protocol.rs b/crates/chia-protocol/src/chia_protocol.rs index a24fe53b8..655a03e92 100644 --- a/crates/chia-protocol/src/chia_protocol.rs +++ b/crates/chia-protocol/src/chia_protocol.rs @@ -146,7 +146,9 @@ pub enum ProtocolMessageTypes { #[cfg(feature = "py-bindings")] impl chia_traits::ChiaToPython for ProtocolMessageTypes { fn to_python<'a>(&self, py: pyo3::Python<'a>) -> pyo3::PyResult> { - Ok(pyo3::IntoPy::into_py(*self, py).bind(py).clone().into_any()) + Ok(pyo3::IntoPyObject::into_pyobject(*self as u8, py)? + .clone() + .into_any()) } } @@ -171,7 +173,9 @@ pub enum NodeType { #[cfg(feature = "py-bindings")] impl chia_traits::ChiaToPython for NodeType { fn to_python<'a>(&self, py: pyo3::Python<'a>) -> pyo3::PyResult> { - Ok(pyo3::IntoPy::into_py(*self, py).bind(py).clone().into_any()) + Ok(pyo3::IntoPyObject::into_pyobject(*self as u8, py)? + .clone() + .into_any()) } } diff --git a/crates/chia-protocol/src/coin_spend.rs b/crates/chia-protocol/src/coin_spend.rs index ffbd3c9c1..0cfdd3910 100644 --- a/crates/chia-protocol/src/coin_spend.rs +++ b/crates/chia-protocol/src/coin_spend.rs @@ -23,6 +23,6 @@ impl CoinSpend { // Convert result into potential child class let instance = cls.call1((cs.coin, cs.puzzle_reveal, cs.solution))?; - Ok(instance.into_py(py)) + Ok(instance.into_pyobject(py)?.unbind()) } } diff --git a/crates/chia-protocol/src/lazy_node.rs b/crates/chia-protocol/src/lazy_node.rs index 59b9824e4..8bde2f037 100644 --- a/crates/chia-protocol/src/lazy_node.rs +++ b/crates/chia-protocol/src/lazy_node.rs @@ -1,6 +1,6 @@ use clvmr::{allocator::NodePtr, allocator::SExp, Allocator}; use pyo3::prelude::*; -use pyo3::types::{PyBytes, PyTuple}; +use pyo3::types::PyBytes; use std::rc::Rc; #[pyclass(subclass, unsendable, frozen)] @@ -10,12 +10,6 @@ pub struct LazyNode { node: NodePtr, } -impl ToPyObject for LazyNode { - fn to_object(&self, py: Python<'_>) -> PyObject { - Bound::new(py, self.clone()).unwrap().to_object(py) - } -} - #[pymethods] impl LazyNode { #[getter(pair)] @@ -24,7 +18,7 @@ impl LazyNode { SExp::Pair(p1, p2) => { let r1 = Self::new(self.allocator.clone(), *p1); let r2 = Self::new(self.allocator.clone(), *p2); - let v = PyTuple::new_bound(py, &[r1, r2]); + let v = (r1, r2).into_pyobject(py)?; Ok(Some(v.into())) } SExp::Atom => Ok(None), @@ -34,9 +28,7 @@ impl LazyNode { #[getter(atom)] pub fn atom(&self, py: Python<'_>) -> Option { match &self.allocator.sexp(self.node) { - SExp::Atom => { - Some(PyBytes::new_bound(py, self.allocator.atom(self.node).as_ref()).into()) - } + SExp::Atom => Some(PyBytes::new(py, self.allocator.atom(self.node).as_ref()).into()), SExp::Pair(..) => None, } } diff --git a/crates/chia-protocol/src/program.rs b/crates/chia-protocol/src/program.rs index 45067a97a..71e5787eb 100644 --- a/crates/chia-protocol/src/program.rs +++ b/crates/chia-protocol/src/program.rs @@ -297,9 +297,9 @@ fn clvm_serialize(a: &mut Allocator, o: &Bound<'_, PyAny>) -> PyResult #[cfg(feature = "py-bindings")] fn to_program(py: Python<'_>, node: LazyNode) -> PyResult> { - let int_module = PyModule::import_bound(py, "chia.types.blockchain_format.program")?; + let int_module = PyModule::import(py, "chia.types.blockchain_format.program")?; let ty = int_module.getattr("Program")?; - ty.call1((node.into_py(py),)) + ty.call1((node.into_pyobject(py)?,)) } #[cfg(feature = "py-bindings")] diff --git a/crates/chia-protocol/src/spend_bundle.rs b/crates/chia-protocol/src/spend_bundle.rs index b5ab9dc9b..ffa29c4ad 100644 --- a/crates/chia-protocol/src/spend_bundle.rs +++ b/crates/chia-protocol/src/spend_bundle.rs @@ -102,10 +102,10 @@ impl SpendBundle { ) -> PyResult { let aggregated = Bound::new(py, Self::aggregate(&spend_bundles))?; if aggregated.is_exact_instance(cls) { - Ok(aggregated.into_py(py)) + Ok(aggregated.into_pyobject(py)?.unbind().into_any()) } else { - let instance = cls.call_method1("from_parent", (aggregated.into_py(py),))?; - Ok(instance.into_py(py)) + let instance = cls.call_method1("from_parent", (aggregated.into_pyobject(py)?,))?; + Ok(instance.into_pyobject(py)?.unbind().into_any()) } } @@ -122,7 +122,7 @@ impl SpendBundle { None, )?; - Ok(instance.into_py(py)) + Ok(instance.into_pyobject(py)?.unbind()) } #[pyo3(name = "name")] diff --git a/crates/chia-protocol/src/wallet_protocol.rs b/crates/chia-protocol/src/wallet_protocol.rs index 731edc490..29c5425d4 100644 --- a/crates/chia-protocol/src/wallet_protocol.rs +++ b/crates/chia-protocol/src/wallet_protocol.rs @@ -300,7 +300,9 @@ pub enum RejectStateReason { #[cfg(feature = "py-bindings")] impl chia_traits::ChiaToPython for RejectStateReason { fn to_python<'a>(&self, py: pyo3::Python<'a>) -> pyo3::PyResult> { - Ok(pyo3::IntoPy::into_py(*self, py).bind(py).clone()) + Ok(pyo3::IntoPyObject::into_pyobject(*self as u8, py)? + .clone() + .into_any()) } } @@ -318,7 +320,9 @@ pub enum MempoolRemoveReason { #[cfg(feature = "py-bindings")] impl chia_traits::ChiaToPython for MempoolRemoveReason { fn to_python<'a>(&self, py: pyo3::Python<'a>) -> pyo3::PyResult> { - Ok(pyo3::IntoPy::into_py(*self, py).bind(py).clone()) + Ok(pyo3::IntoPyObject::into_pyobject(*self as u8, py)? + .clone() + .into_any()) } } diff --git a/crates/chia-traits/src/from_json_dict.rs b/crates/chia-traits/src/from_json_dict.rs index e15f78289..0bb601ffe 100644 --- a/crates/chia-traits/src/from_json_dict.rs +++ b/crates/chia-traits/src/from_json_dict.rs @@ -51,7 +51,7 @@ where { fn from_json_dict(o: &Bound<'_, PyAny>) -> PyResult { let mut ret = Vec::::new(); - for v in o.iter()? { + for v in o.try_iter()? { ret.push(::from_json_dict(&v?)?); } Ok(ret) diff --git a/crates/chia-traits/src/int.rs b/crates/chia-traits/src/int.rs index 2b49e155c..7c710a2c7 100644 --- a/crates/chia-traits/src/int.rs +++ b/crates/chia-traits/src/int.rs @@ -10,9 +10,9 @@ macro_rules! primitive_int { ($t:ty, $name:expr) => { impl ChiaToPython for $t { fn to_python<'a>(&self, py: Python<'a>) -> PyResult> { - let int_module = PyModule::import_bound(py, "chia_rs.sized_ints")?; + let int_module = PyModule::import(py, "chia_rs.sized_ints")?; let ty = int_module.getattr($name)?; - ty.call1((self.into_py(py),)) + ty.call1((self.into_pyobject(py)?.into_any(),)) } } }; @@ -40,7 +40,7 @@ impl ChiaToPython for Option { impl ChiaToPython for Vec { fn to_python<'a>(&self, py: Python<'a>) -> PyResult> { - let ret = PyList::empty_bound(py); + let ret = PyList::empty(py); for v in self { ret.append(v.to_python(py)?)?; } @@ -50,32 +50,32 @@ impl ChiaToPython for Vec { impl ChiaToPython for bool { fn to_python<'a>(&self, py: Python<'a>) -> PyResult> { - Ok(PyBool::new_bound(py, *self).as_any().clone()) + Ok(PyBool::new(py, *self).as_any().clone()) } } impl ChiaToPython for String { fn to_python<'a>(&self, py: Python<'a>) -> PyResult> { - Ok(PyString::new_bound(py, self.as_str()).into_any()) + Ok(PyString::new(py, self.as_str()).into_any()) } } impl ChiaToPython for (T, U) { fn to_python<'a>(&self, py: Python<'a>) -> PyResult> { - Ok(PyTuple::new_bound(py, [self.0.to_python(py)?, self.1.to_python(py)?]).into_any()) + Ok(PyTuple::new(py, [self.0.to_python(py)?, self.1.to_python(py)?])?.into_any()) } } impl ChiaToPython for (T, U, V) { fn to_python<'a>(&self, py: Python<'a>) -> PyResult> { - Ok(PyTuple::new_bound( + Ok(PyTuple::new( py, [ self.0.to_python(py)?, self.1.to_python(py)?, self.2.to_python(py)?, ], - ) + )? .into_any()) } } diff --git a/crates/chia-traits/src/to_json_dict.rs b/crates/chia-traits/src/to_json_dict.rs index 48e60a0b4..789ef80ac 100644 --- a/crates/chia-traits/src/to_json_dict.rs +++ b/crates/chia-traits/src/to_json_dict.rs @@ -1,5 +1,6 @@ use pyo3::prelude::*; use pyo3::types::PyList; +use pyo3::BoundObject; pub trait ToJsonDict { fn to_json_dict(&self, py: Python<'_>) -> PyResult; @@ -9,7 +10,7 @@ macro_rules! to_json_primitive { ($t:ty) => { impl $crate::to_json_dict::ToJsonDict for $t { fn to_json_dict(&self, py: Python<'_>) -> pyo3::PyResult { - Ok(self.to_object(py)) + Ok(self.into_pyobject(py)?.into_any().unbind()) } } }; @@ -30,7 +31,7 @@ to_json_primitive!(String); impl ToJsonDict for Vec { fn to_json_dict(&self, py: Python<'_>) -> PyResult { - let list = PyList::empty_bound(py); + let list = PyList::empty(py); for v in self { list.append(v.to_json_dict(py)?)?; } @@ -50,7 +51,7 @@ impl ToJsonDict for Option { // if we need more of these, we should probably make a macro impl ToJsonDict for (T, U) { fn to_json_dict(&self, py: Python<'_>) -> PyResult { - let list = PyList::empty_bound(py); + let list = PyList::empty(py); list.append(self.0.to_json_dict(py)?)?; list.append(self.1.to_json_dict(py)?)?; Ok(list.into()) @@ -59,7 +60,7 @@ impl ToJsonDict for (T, U) { impl ToJsonDict for (T, U, W) { fn to_json_dict(&self, py: Python<'_>) -> PyResult { - let list = PyList::empty_bound(py); + let list = PyList::empty(py); list.append(self.0.to_json_dict(py)?)?; list.append(self.1.to_json_dict(py)?)?; list.append(self.2.to_json_dict(py)?)?; diff --git a/package-lock.json b/package-lock.json index 5c2560682..c171ba057 100644 --- a/package-lock.json +++ b/package-lock.json @@ -5,15 +5,14 @@ "packages": { "": { "devDependencies": { - "prettier": "^3.4.1" + "prettier": "^3.4.2" } }, "node_modules/prettier": { - "version": "3.4.1", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.4.1.tgz", - "integrity": "sha512-G+YdqtITVZmOJje6QkXQWzl3fSfMxFwm1tjTyo9exhkmWSqC4Yhd1+lug++IlR2mvRVAxEDDWYkQdeSztajqgg==", + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.4.2.tgz", + "integrity": "sha512-e9MewbtFo+Fevyuxn/4rrcDAaq0IYxPGLvObpQjiZBMAzB9IGmzlnG9RZy3FFas+eBMu2vA0CszMeduow5dIuQ==", "dev": true, - "license": "MIT", "bin": { "prettier": "bin/prettier.cjs" }, diff --git a/package.json b/package.json index a13659367..474b8042e 100644 --- a/package.json +++ b/package.json @@ -5,6 +5,6 @@ "fmt": "prettier --write ." }, "devDependencies": { - "prettier": "^3.4.1" + "prettier": "^3.4.2" } } diff --git a/wheel/src/api.rs b/wheel/src/api.rs index eaeca3210..6cc98f1f5 100644 --- a/wheel/src/api.rs +++ b/wheel/src/api.rs @@ -90,10 +90,7 @@ pub fn compute_merkle_set_root<'p>( use pyo3::types::PyBytesMethods; buffer.push(b.as_bytes().try_into()?); } - Ok(PyBytes::new_bound( - py, - &compute_merkle_root_impl(&mut buffer), - )) + Ok(PyBytes::new(py, &compute_merkle_root_impl(&mut buffer))) } #[pyfunction] @@ -180,8 +177,8 @@ pub fn get_puzzle_and_solution_for_coin<'a>( }; */ Ok(( - PyBytes::new_bound(py, &serialize(&allocator, puzzle)?), - PyBytes::new_bound(py, &serialize(&allocator, solution)?), + PyBytes::new(py, &serialize(&allocator, puzzle)?), + PyBytes::new(py, &serialize(&allocator, solution)?), )) } @@ -240,7 +237,7 @@ type CoinSpendRef = (Coin, PyBackedBytes, PyBackedBytes); fn convert_list_of_tuples(spends: &Bound<'_, PyAny>) -> PyResult> { let mut native_spends = Vec::::new(); - for s in spends.iter()? { + for s in spends.try_iter()? { let s = s?; let tuple = s.downcast::()?; let coin = tuple.get_item(0)?.extract::()?; @@ -257,7 +254,7 @@ fn solution_generator<'p>( spends: &Bound<'_, PyAny>, ) -> PyResult> { let spends = convert_list_of_tuples(spends)?; - Ok(PyBytes::new_bound(py, &native_solution_generator(spends)?)) + Ok(PyBytes::new(py, &native_solution_generator(spends)?)) } #[pyfunction] @@ -266,7 +263,7 @@ fn solution_generator_backrefs<'p>( spends: &Bound<'_, PyAny>, ) -> PyResult> { let spends = convert_list_of_tuples(spends)?; - Ok(PyBytes::new_bound( + Ok(PyBytes::new( py, &native_solution_generator_backrefs(spends)?, )) @@ -402,7 +399,7 @@ fn fast_forward_singleton<'p>( let solution = node_from_bytes(&mut a, spend.solution.as_slice())?; let new_solution = native_ff(&mut a, puzzle, solution, &spend.coin, new_coin, new_parent)?; - Ok(PyBytes::new_bound( + Ok(PyBytes::new( py, node_to_bytes(&a, new_solution)?.as_slice(), ))