-
Notifications
You must be signed in to change notification settings - Fork 81
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
Refactor: Drop typestate pattern for building sockets #468
Conversation
ee70adc
to
2d5c1b7
Compare
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.
Thanks for the great write-up. I think I agree, and I also think I was the one arguing hardest for not requiring new users to think about channels, but like you say I guess they still do, due to the generic parameter.
Also, the situation has changed a bit, and I think a big win for this is also that it's easier to maintain and understand for new contributors coming to the library itself.
@garryod what do you think?
Regarding "
I would go with that approach myself if I were designing this library, mainly because ggrs is very clear that it expects unreliable transport and the various "make a socket/channel for ggrs" helper functions are a) trivial and b) initially made me think that there might be something non-trivial involved in making a ggrs socket/channel. But it is a larger and more opinionated change so I held off doing it for this PR's first cut. We could also go further and have |
Having thought about it, I agree on all points. Let's drop the implementation on the socket and just have a runtime warning if the channel is not configured as unreliable |
Many of the ci failures are unrelated, but fixed now. Feel free to either rebase or merge main into this branch. |
The typestate pattern caused a lot of boilerplate w.r.t. both the implementation and the usage of matchbox socket. There appeared to be only three actual uses of the typestate pattern: * Preventing client code from calling build() without adding a channel. * Some convenience methods to read/write from the first channel in the case where where was only one channel. * Making sure that ggrs::NonBlockingSocket is only implemented for WebRtcSocket in the case where it is a single channel socket, to avoid the case where a reliable socket is accidentally used as a GGRS socket (or a unreliable socket is used for GGRS + some other data). The first and (mostly) the last points can be accomplished at runtime via asserts, and owing to the "set it up and use it from one place" nature of this library I think it's highly unlikely that having a runtime assertion is problematic. This is a breaking change owing to the removal of the SingleChannel-variant convenience methods, but it should be trivially easy to migrate by looking at modified example code.
343e8a9
to
39b4e34
Compare
To create GGRS sockets, users can still create new sockets by specifying ChannelConfig::unreliable() and using WebRtcSocket::take_channel() to detach the channel so it can be owned by GGRS. This makes it clearer that there's nothing special about GGRS sockets.
39b4e34
to
469e694
Compare
I think the remaining ci failures now are actual relevant failures. The bevy signalling server examples need to be updated.
Otherwise, this looks good to merge :) |
Compile errors should be fixed now, hopefully (you correctly guessed that I had forgotten to test & lint with the --all-targets flag and the signaling feature flag). |
Formatting fixed, sorry. Please re-run CI and this time it should actually go all green. (Looking forward to - I assume - being able to trigger actions workflows myself once I am no longer a first time contributor..) |
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.
Looks good! thanks for all the good work on the whole stack
The typestate pattern caused a lot of boilerplate w.r.t. both the implementation and the usage of matchbox socket.
There appeared to be only three actual uses of the typestate pattern:
WebRtcSocket in the case where it is a single channel socket, to avoid the case where a reliable socket is accidentally used as a GGRS socket (or a unreliable socket is used for GGRS + some other data).
The first and (mostly) the last points can be accomplished at runtime via asserts, and owing to the "set it up and use it from one place" nature of this library I think it's highly unlikely that having a runtime assertion is problematic.
This is a breaking change owing to the removal of the SingleChannel-variant convenience methods, but it should be trivially easy to migrate by looking at modified example code.
This is definitely a "change based on personal taste", so I understand if your personal preferences do not align and you do not want to go down this route. But since I've gone to the trouble of writing this PR/micro-essay and you're reading it, well, you may as well keep reading, right? :)
TLDR: In my opinion, the benefits of the typestate pattern here are not justifying its cost.
As a new user approaching this library, one of my stumbling blocks was "why is there this ChannelPlurality generic that's front and center?". Not "what is it" - I understand the idea of making bad states unrepresentable, I'm already familiar with the typestate pattern in particular, I've used Rust plenty and I've shipped prod code with type wizardry in Scala, Typescript, etc - but rather the why, as in what motivates it.
So, I spent a while diving through the codebase to figure out what makes a single channel webrtc socket inherently different to a many channel socket in some way that makes it worth modeling it at a type system level - and after going through the exercise of removing the typestate pattern (I find editing code & following the compile errors a good way to understand it), all I really found was the convenience methods for a single channel WebRtcSocket - and (IMO) these more obscure how the library is structured than aid in understanding it.
So I figured I'd raise my changes as a PR to remove the typestate pattern and see what others thought - the result is at least less boilerplate-y.
I did read #160 to understand the thinking behind the initial introduction of the typestate pattern, and it seems to boil down to "let's make it easier for new users". I think that's a great goal, but I would argue that a somewhat-vaguely-named (and possibly misleading, at least for me) generic parameter is less easy (or at least, less simple) than embracing the concept of channels; the updated example code in this PR shows that dealing with a single channel is straightforward enough.
(I figure backwards compatibility may also have been a concern that motivated the introduction of the convenience methods I mentioned in my second bullet above? But I think the changes from dealing with channels are easy enough to adjust to - a single
.channel_mut(0)
in the (probably) 2 places where a single-channel user would be reading/writing from/to the socket would be the sum total of it.)Closes: #332