Skip to content
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

QueryState::new_with_state #6240

Closed
wants to merge 1 commit into from

Conversation

Suficio
Copy link
Contributor

@Suficio Suficio commented Oct 11, 2022

Objective

Allows providing state to QueryState which requires input other than WorldQuery::init_state(&mut World) to be constructed.

This allows the user to provide a custom state such as ComponentId for dynamic queries which cannot be derived from the type definition.

Solution

Create new method QueryState::new_with_state, which is now called by QueryState::new using WorldQuery::init_state.

@alice-i-cecile alice-i-cecile added A-ECS Entities, components, systems, and events C-Usability A targeted quality-of-life change that makes Bevy easier to use labels Oct 11, 2022
Copy link
Member

@alice-i-cecile alice-i-cecile left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The example is great, but we probably also want a third query_state initialization showing how to init filter state.

@alice-i-cecile
Copy link
Member

Awesome work; thanks for the prompt response. This is a niche feature, but I think it will be useful as part of our "dynamic components" story.

@Suficio
Copy link
Contributor Author

Suficio commented Oct 11, 2022

Awesome work; thanks for the prompt response. This is a niche feature, but I think it will be useful as part of our "dynamic components" story.

No problem, your prompt response is much more appreciated :)

With this feature it is already possible to make a dynamic query by implementing WorldQuery on some Ptr type.

Copy link
Member

@zicklag zicklag left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not qualified enough to review the Bevy query internals yet/anymore, but I seem to remember needing this in my now very old scripting API/PR, so.., 👍 😁

@alice-i-cecile
Copy link
Member

Yep, I mostly wanted to hear if you thought it was useful :) The implementation is trivial, and the docs are great.

@BoxyUwU BoxyUwU self-requested a review October 17, 2022 02:05
@BoxyUwU
Copy link
Member

BoxyUwU commented Oct 17, 2022

I expect this is incredibly unsound as-is since it would let you construct a QueryState<&T> where the state has a componentid for U, which would then get bevy to turn a pointer to U into a &T.

Having a way to do the goal of this PR does seem useful for dynamic queries and I think relations could make use of this too.

@Suficio
Copy link
Contributor Author

Suficio commented Oct 21, 2022

Nice catch @BoxyUwU!

While there is nothing unsafe that this function does, indeed we can incorrectly cast types as a consequence. There are two ways I see that this can be resolved.

  1. Validate ComponentId for &T queries during <&T as WorldQuery>::init_fetch
  2. Make QueryState::new_with_state unsafe

@alice-i-cecile
Copy link
Member

The former is definitely safer. This would be a one time cost paid during system / query initialization, correct?

@BoxyUwU
Copy link
Member

BoxyUwU commented Oct 21, 2022

no, Fetch is created per archetype/table (which one depends on whether it's a dense or sparse iteration)

@Suficio
Copy link
Contributor Author

Suficio commented Oct 23, 2022

Made the method unsafe and added the appropriate safety comment

@jakobhellermann
Copy link
Contributor

Made the method unsafe and added the appropriate safety comment

Currently the # Safety comment only shows a bad example and suggest using safer methods instead, usually these comments also contain an exhautive list of things you need to validate so that the call is guaranteed safe. Something like

/// # Safety
/// - index must be in bounds
/// - the memory in the [`BlobVec`] starting at index `index`, of a size matching this [`BlobVec`]'s
/// `item_layout`, must have been previously allocated.


As I understand it, the point of the new_with_state method is to be able to be used if you only know a few ComponentIds at runtime and want to get Ptr<'_>s to them, instead of specifying T to get &Ts, right? And the reason you can't just

struct ComponentIdQuery(ComponentId);
struct ComponentIdQueryMut(ComponentId);

impl WorldQuery for ComponentIdQuery {
  type Item = Ptr<'_>;
  ...
}
impl WorldQuery for ComponentIdQueryMut {
  type Item = (PtrMut<'_>, &mut Ticks);
  ...
}

is that currently init_state is called just with the &mut World without access to any runtime query info.
What if instead we add a type Config to WorldQuery?

trait WorldQuery: WorldQueryGats {
  type State;
+ type Config;
  
-    fn init_state(world: &mut World) -> Self::State;
+    fn init_state(world: &mut World, config: Self::Config) -> Self::State;
}

where the current implementors set Config = () and QueryState::new only works with T: WorldQuery<Config = ()> and we add a fn WorldQuery::new_with_config?

Then you can add impls of WorldQuery for ComponentIdQuery with Item = Ptr<'_> Config = ComponentId and for Vec<T: WorldQuery> with Item = Vec<T::Item> Config = Vec<T::Config>, and you can safely call

let query = QueryState<Vec<ComponentIdQuery>>::new_with_config(vec![my_component_id_1, my_component_id_2]);

@Suficio
Copy link
Contributor Author

Suficio commented Oct 24, 2022

Currently the # Safety comment only shows a bad example and suggest using safer methods instead, usually these comments also contain an exhautive list of things you need to validate so that the call is guaranteed safe.

It's sort of difficult because this method is only unsafe when querying component references, but I reordered the safety comment to more strongly word the requirements for safety.

As I understand it, the point of the new_with_state method is to be able to be used if you only know a few ComponentIds at runtime and want to get Ptr<'_>s to them, instead of specifying T to get &Ts, right?

This is exactly right! However, this isn't the only use, one could likely implement world query for Vec<T>.

Your solution is definitely an improvement, however, I would like to prioritize minimal changes with this PR.

Dynamic query work is still up for discussion on how it should be implemented, and I would like to avoid establishing any future design directions, and instead, work with the API surface as it is available today.

@Suficio Suficio force-pushed the query-state-with-state branch from e104365 to 6cbbb9d Compare October 24, 2022 15:23
@BoxyUwU
Copy link
Member

BoxyUwU commented Oct 24, 2022

It's sort of difficult because this method is only unsafe when querying component references

I disagree with this, worldquery impls should be able to rely on the state being created from their own init_state fn and nothing else. More than just "unsafe" it should be unsound to call new_with_state with a value that isnt from init_state unless the worldquery explicitly relaxes this in its documentation. (if this isnt true the soundness of worldquery code gets a lot trickier to reason about, or we start wanting to newtype all our state to prevent public construction neither of which seems desirable to me)

If the above is true then this API is also kind of Not Great, I would much prefer the solution @jakobhellermann proposes with a type Config; since we could probably make that API safe

@Suficio
Copy link
Contributor Author

Suficio commented Oct 26, 2022

The problem I see is that WorldQuery tuple implementations will require all elements to have Config = (), due to system params requiring Config = () (as systems are not able to provide the config), making queries with arbitrary Config unusable for the majority of use patterns.

I've tried the implementation suggested and it results in a lot of extra type constraints that need to be managed across the codebase, and due to the above, severely restricts the reusability of existing implementations.

More than just "unsafe" it should be unsound to call new_with_state with a value that isnt from init_state unless the worldquery explicitly relaxes this in its documentation.

Due to this restriction on Config = (), I've slightly relaxed the safety comment.

While I agree that queries should be able to rely on their init_state, it is not possible to initialize state using only &World, and the suggested Config approach is not able to facilitate the use cases I need.

@jakobhellermann
Copy link
Contributor

Yeah, the Config = () issue for tuples is one I stumbled upon as well when protoyping my suggestion. So instead I required Config: Default, which needs to propagated to a few places. I then ran into an issue changing the tuple impl and didn't finish the prototype, but when I do, I'll open it as a draft PR to have a discussion on whether the few Config: Default bounds are too annoying.

@Suficio
Copy link
Contributor Author

Suficio commented Oct 26, 2022

I think one could argue that having a default here is equally dangerous as providing the state outright, due to the potential implications of accidentally using a ComponentId default for Ptrs.

It results in a situation where you don't know whether init_state config has correct data or whether it needs to be inferred otherwise. I guess we could use an Option but this is still added type complexity for a very niche benefit that this PR brings.

The added benefit of this PR would be that you can avoid the added complication of manipulating type bounds.

@jakobhellermann
Copy link
Contributor

I posted a PR of my proposal: #6390

@alice-i-cecile alice-i-cecile added the X-Controversial There is active debate or serious implications around merging this PR label Oct 27, 2022
@Suficio Suficio force-pushed the query-state-with-state branch from ae570b0 to 767efee Compare June 12, 2023 22:10
Improved doc comment

Further improved doc

Make QueryState::new_with_state unsafe

Reorder safety comment

Relax safety comment
@Suficio Suficio force-pushed the query-state-with-state branch from 767efee to d3dfbfa Compare June 12, 2023 22:12
@Suficio
Copy link
Contributor Author

Suficio commented Jan 18, 2024

Resolved by #9774

@Suficio Suficio closed this Jan 18, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-ECS Entities, components, systems, and events C-Usability A targeted quality-of-life change that makes Bevy easier to use X-Controversial There is active debate or serious implications around merging this PR
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants