Skip to content

Commit f753790

Browse files
committed
guide: note existence of PyFunction::new_closure
1 parent 86a1116 commit f753790

File tree

2 files changed

+8
-15
lines changed

2 files changed

+8
-15
lines changed

guide/src/function.md

+6-13
Original file line numberDiff line numberDiff line change
@@ -197,12 +197,6 @@ Docstring: This function adds two unsigned 64-bit integers.
197197
Type: builtin_function_or_method
198198
```
199199

200-
### Closures
201-
202-
Currently, there are no conversions between `Fn`s in Rust and callables in Python. This would
203-
definitely be possible and very useful, so contributions are welcome. In the meantime, you can do
204-
the following:
205-
206200
### Calling Python functions in Rust
207201

208202
You can pass Python `def`'d functions and built-in functions to Rust functions [`PyFunction`]
@@ -217,13 +211,12 @@ with only positional args.
217211

218212
### Calling Rust functions in Python
219213

220-
If you have a static function, you can expose it with `#[pyfunction]` and use [`wrap_pyfunction!`]
221-
to get the corresponding [`PyCFunction`]. For dynamic functions, e.g. lambdas and functions that
222-
were passed as arguments, you must put them in some kind of owned container, e.g. a `Box`.
223-
(A long-term solution will be a special container similar to wasm-bindgen's `Closure`). You can
224-
then use a `#[pyclass]` struct with that container as a field as a way to pass the function over
225-
the FFI barrier. You can even make that class callable with `__call__` so it looks like a function
226-
in Python code.
214+
The ways to convert a Rust function into a Python object vary depending on the function:
215+
216+
- Named functions, e.g. `fn foo()`: add `#[pyfunction]` and then use [`wrap_pyfunction!`] to get the corresponding [`PyCFunction`].
217+
- Anonymous functions (or closures), e.g. `foo: fn()` either:
218+
- use a `#[pyclass]` struct which stores the function as a field and implement `__call__` to call the stored function.
219+
- use `PyFunction::new_closure` to create an object directly from the function.
227220

228221
[`PyAny::is_callable`]: {{#PYO3_DOCS_URL}}/pyo3/struct.PyAny.html#tymethod.is_callable
229222
[`PyAny::call`]: {{#PYO3_DOCS_URL}}/pyo3/struct.PyAny.html#tymethod.call

guide/src/function/error_handling.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -169,14 +169,14 @@ fn parse_int(s: String) -> PyResult<usize> {
169169

170170
The Rust compiler will not permit implementation of traits for types outside of the crate where the type is defined. (This is known as the "orphan rule".)
171171

172-
Given a type `OtherError` which is defined in thirdparty code, there are two main strategies available to integrate it with PyO3:
172+
Given a type `OtherError` which is defined in third-party code, there are two main strategies available to integrate it with PyO3:
173173

174174
- Create a newtype wrapper, e.g. `MyOtherError`. Then implement `From<MyOtherError> for PyErr` (or `PyErrArguments`), as well as `From<OtherError>` for `MyOtherError`.
175175
- Use Rust's Result combinators such as `map_err` to write code freely to convert `OtherError` into whatever is needed. This requires boilerplate at every usage however gives unlimited flexibility.
176176

177177
To detail the newtype strategy a little further, the key trick is to return `Result<T, MyOtherError>` from the `#[pyfunction]`. This means that PyO3 will make use of `From<MyOtherError> for PyErr` to create Python exceptions while the `#[pyfunction]` implementation can use `?` to convert `OtherError` to `MyOtherError` automatically.
178178

179-
The following example demonstrates this for some imaginary thirdparty crate `some_crate` with a function `get_x` returning `Result<i32, OtherError>`:
179+
The following example demonstrates this for some imaginary third-party crate `some_crate` with a function `get_x` returning `Result<i32, OtherError>`:
180180

181181
```rust
182182
# mod some_crate {

0 commit comments

Comments
 (0)