-
Notifications
You must be signed in to change notification settings - Fork 2.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Updated STL type hints to use collections.abc #5566
base: master
Are you sure you want to change the base?
Conversation
The failing check is unrelated, I think (maybe rerun is enough). |
After seeing #5498 and digging deeper into the caster code, I noticed that I have to think more about this. These three functions restrict the casters further than I thought: pybind11/include/pybind11/stl.h Lines 88 to 119 in 2e382c0
For example, it requires the mapping to be of type or subtype of set or frozenset or have dict_keys if I understand correctly.The caster itself uses the Mapping protocol though and could easily be changed to fully allow it. I will add some tests to better map out what is allowed and what not and how this relates to the type hints. Git blame directed me to #4686, which seems to have more insight. |
@timohl Did you see these already?
I'm fine if you want to work on those functions. I'm thinking it's best to keep the current logic, which is super fast, but where we're currently returning |
include/pybind11/stl.h
Outdated
if (!convert) { | ||
return false; | ||
} | ||
if (!(isinstance(src, module_::import("collections.abc").attr("Set")) | ||
&& hasattr(src, "__contains__") && hasattr(src, "__iter__") | ||
&& hasattr(src, "__len__"))) { | ||
return false; | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Did you consider moving this code into PyObjectTypeIsConvertibleToStdSet()
?
Then it would be easier to use same logic in other custom type casters.
Giving PyObjectTypeIsConvertibleToStdSet()
the additional bool convert
argument seems fine to me.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I integrated the checks, but had to change the argument from PyObject*
to const handle &
in order to use isinstance
and hasattr
.
Is that ok, or should I use PyObject based functions from the Python C API directly?
Honestly, I do not like how obscure those functions currently are due to all those checks.
While there is a comment explaining some intentions, I think it could be made clearer what should pass and what not.
I will add and improve comments later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is that ok
Yes. — The functions came from PyCLIF, and I wanted to keep them compatible, but that's not a concern anymore.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I only glanced through very quickly. Is this still in draft mode intentionally?
inline bool PyObjectTypeIsConvertibleToStdVector(PyObject *obj) { | ||
if (PySequence_Check(obj) != 0) { | ||
return !PyUnicode_Check(obj) && !PyBytes_Check(obj); | ||
inline bool HandleIsConvertibleToStdVector(const handle &src) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What do you think about these names?
object_is_convertible_to_std_vector
object_is_convertible_to_std_set
object_is_convertible_to_std_map
(handle
happens to be the argument type here, but that's really secondary, or even subject to change.)
I would like to improve the comments before finalizing and being ready to merge. Also, your comment about the function names sounds good. I will change that. |
No rush, at all, from my end. I just wanted to be sure you're not waiting for my feedback. |
Would it also be possible to update |
Going through the code here: pybind11/include/pybind11/cast.h Lines 1289 to 1392 in f365314
There are a bunch of other types that could be changed:
|
Currently stub generators typically know that |
Description
This updates type hints for the
set
,map
,list
andarray
STL casters to use the more genericcollections.abc
types in convertible arguments.set_caster
collections.abc.Set
set
map_caster
collections.abc.Mapping
dict
list_caster
collections.abc.Sequence
list
array_caster
collections.abc.Sequence
list
For
map_caster
this is exactly how the caster works.Unfortunately,
list_caster
,set_caster
andarray_caster
work a bit different regardingnoconvert
:For args
list_caster
andarray_caster
always allowsequence
andset_caster
always allowsanyset
, but all three allowiterable
in convert mode.The current system only differs between input and output type (
io_name
) and falls back to the output type in noconvert args.These casters' behavior would require three different type hints: arg-convert, arg-noconvert, return.
Therefore, I currently see no way to improve these type hints further without deeper changes.
So for now, I think this should be a good compromise for most use cases.
Additionally, the
array_caster
was updated to match thetyping.Annotated
style of numpy/eigen type hints.Suggested changelog entry: