-
Notifications
You must be signed in to change notification settings - Fork 450
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
test marker traits more rigorously on public API types #756
Comments
Can I take a jab at implementing these tests? If it's ok, can you give me some pointers on where I can find similar tests? |
I think Ideally there would be some way to write a test (maybe using shell script or Python, not sure) that would fail in the case of a public API type that doesn't have an oibit test. That way, if a new public API type is added, we're forced to add an oibit test for it. But maybe this is too much machinery. |
Thank you for the quick response! So, if I understood correctly, we want to avoid having to repeat groups of assertions like: assert_send::<Regex>();
assert_sync::<Regex>();
assert_unwind_safe::<Regex>();
assert_ref_unwind_safe::<Regex>();
assert_send::<RegexBuilder>();
assert_sync::<RegexBuilder>();
assert_unwind_safe::<RegexBuilder>();
assert_ref_unwind_safe::<RegexBuilder>(); by writing a script that takes a list (or somehow knows about every public API type) and generates the list of assertions? Sounds good to me, and it is very interesting. Let me know if this is the case. |
While avoiding repetition is a good idea, that should probably be done with a macro. The purpose of the script is not about repetition. It's about automation. Consider the case of when a new public API type is added. How does that person know to go add it to the oibit traits? They don't. It would be good to have a test that fails if a public API type exists and is not in the oibit test. With that said, I do not want anything that is terribly complex, difficult to maintain or brings in new dependencies. If there isn't a simple solution for it, then it might not be worth it. If you do want to pursue this extra automation step, then please discuss your plan with me before you invest too much effort. But the main objective is to get oibit tests for each public API type. |
Yeap, I agree. I also thought about a macro but wasn't sure. 😅 Ok, so, then this is divided into two tasks: Tasks
Constraints for the automation:
IdeasMy idea for task 2 right now is to check how does documentation for types gets generated, and maybe somehow extract the types from there? I'm pretty new to Rust so I don't know what introspection abilities it has. I think task 3 might be avoided by doing something like: fn assert_oibits<T: Send + Sync + UnwindSafe + RefUnwindSafe>() {}
assert_oibits::<Regex>(); Let me know if all of this makes sense. |
What I got for task 2 so far is:
Point 2 seems way overcomplicated. Points 2 and 3 add dependencies. So, I think the best one, right now, is point 1. |
Combining the marker traits into one routine is a good approach. I think all public API types in this crate should satisfy all four of those? Hmm, maybe not, the iterators might not. Ideally, |
@BurntSushi you might be able to get something to work with |
@jyn514 Thank you! @alexfertel perhaps give that a whirl? |
See also the ugly hack tokio uses (and the discussion about showing this in rustdoc): rust-lang/rfcs#2963 (comment) |
Yeap, I'll give it a try! |
@BurntSushi Yeap, I managed to get the public types from the generated JSON. 😁 🎉 I'll try to have a Python script working in the next couple of days, and I'll submit a PR. Would that be ok? |
SGTM! Thanks. :) |
@BurntSushi I will use the function fn assert_oibits<T: Send + Sync + UnwindSafe + RefUnwindSafe>() {} |
Ah that's great. Makes it easy. I suppose it would be good to switch from |
Well, scratch that last comment, I generated the tests and some of the types aren't My idea was to check all the traits for every type, but now it's not so easy. We're not solving the problem of "Having a test that fails if a public API type exists and is not in the oibit test", because each time we run the script, the tests have to be generated based on the traits implemented by the types, not just the types. The solution I see is the following: The first time we generate the tests, we maintain a map of the traits implemented by the types. We can only insert to this map if there are new types exposed by the crate. Do you see another way of solving this? |
@alexfertel I think the key simplification here is that we don't need a robust script to generate the tests. Technically, writing them by hand would be just fine. What we need is way to check for the existence of a test. Since we control what the tests look like, that seems solvable with |
Since the
regex
crate uses interior mutability and a bit of synchronization, it follows that marker traits likeSend
,Sync
,UnwindSafe
andRefUnwindSafe
are quite relevant and can change based on internal implementation details. #749 for example, removed thethread_local
dependency and got a bit more rigorous about these marker traits on the mainRegex
/RegexSet
types, but stopped there.It turns out other types are impacted too. @kangalioo noted that, for example,
Send
was not implemented on theCaptureMatches
iterator inregex 1.3.9
, but it is now implemented inregex 1.4.5
. I'm actually not sure why it wasn't implemented before. Theregex
crate doesn't use any non-Send stuff, so I'm not sure where that was coming from. I haven't investigated yet.Either way, this is a compatibility hazard. Going from
!Send
toSend
is fine, but it seems just as likely that we could regress too. The only way I can think to solve this is to assert expected marker traits for all public API types. It seems kind of absurd and tedious, but I don't have any better ideas.The text was updated successfully, but these errors were encountered: