x/net/websocket: remedy neglect; merge with gorilla websocket?
zombiezen opened this issue · 63 comments
It is not well maintained, and essentially everyone uses https://github.com/gorilla/websocket.
This was raised as a concern in #17244: because it is in the x/net repository, people attach greater value to it than it really ought to have.
/cc @dsnet
EDIT 2016-12-02: Copying in my better-phrased problem statement from later in the thread:
The problem I see is that the ownership/maintenance story of x/net/websocket is unclear. It may suit your needs, and that is fine! However, it's not getting the same level of support as the standard library or even gorilla/websocket; it has no established owner AFAIK.
Along the lines of #17244, there is concern that it elevates the status and visibility of the package. I've anecdotally heard of it confusing many new users whose needs were much better filled with gorilla/websocket. I have not heard of someone using gorilla/websocket and then deciding to use x/net/websocket instead.
We can't just delete something and break people who aren't vendoring it.
We can document that it's deprecated perhaps.
Or somebody could maintain it.
Deprecated might be fine, but my experience is that I (and other devs I've talked to) have wasted time trying to use the package only to find out it fails in production and that Gorilla's package works.
I support a strong deprecation message. Would that message endorse the Gorilla implementation?
If we implement golang/lint#238, I think that will help people avoid wasting time developing with deprecated packages or features.
"fails in production" -- what are the open bugs? There's not much to WebSockets. I'm afraid we're going to spend more time discussing things on this bug than we would just fixing any such issues.
Beyond any bugs that could be fixed by maintenance, the deeper problem with x/net/websocket is that it has the wrong API. It puts an io.Reader/Writer interface on top of what is fundamentally a message-based protocol.
I agree it should be documented as deprecated in favor of gorilla/websocket.
It's been long enough since I last tried to use it, but off-hand, you cannot implement a correct websocket reverse proxy with the current API, since it tries to implement io.Reader/io.Writer. Websocket, for better or for worse, is a frame-oriented protocol, not a stream-oriented protocol.
I agree it has the wrong API. But we could add ReadMessage and WriteMessage to it and deprecate Read and Write.
I am in favor of maintaining less code, though, (not that we ever maintained x/net/websocket much?), and I have no personal or sentimental connection to x/net/websocket, and we actually use gorilla/websocket in Camlistore these days, but it still feels core enough that we should have an official answer for websockets in x/net.
I suppose this all kinda depends on the policy for x/* we figure out in #17244.
Please do not deprecate it. At work, we are using golang.org/x/net/websocket exclusively and we have never had any problem with it*. Quite the opposite, in our use case it's more robust than what we've ever expected.
*: Of course, that says nothing whatsoever about any trouble anyone else might have with it.
Yep. But I guess it means "no more bug fixes for you, ever".
No, not really. Deprecated is not black & white. We continue to fix bugs in other deprecated stuff when there's a good bug report and a clear fix and super minimal risk of breaking existing users. It just means we're not adding features to it, or means that a better way is known to exist.
Great news, honestly. But then again, why to mark it deprecated in the first place? We can cope with it getting no attention from people not loving it, but we do and if we run into a bug we need to resolve, we would be more than happy to contribute the fix.
Status quo anybody?
But then again, why to mark it deprecated in the first place?
To tell users they might be happier elsewhere.
Precisely to @bradfitz's point: Deleting may be too strong, but I want to send a strong signal that there's a much better maintenance story for gorilla/websocket. Effort spent on x/net/websocket may be better spent on gorilla/websocket.
CL https://golang.org/cl/33806 mentions this issue.
To tell users they might be happier elsewhere.
I really really really don't want to sound snarky, but by that logic, the Go repository might be marked deprecated as well - just because users might be happier with Rust, C, Java, ...?
Effort spent on x/net/websocket may be better spent on gorilla/websocket.
I have no intention, nor desire to suggest or propose, regardless how impossible and foolish that would be, to delete github.com/gorilla/websocket, just because we don't and probably never would use it.
I really really really don't want to sound snarky,
Yet you did. And it's not very productive. It's actually just noise in this thread.
If you want to be productive, though, step up and maintain golang.org/x/net/websocket. That's one of the options I listed in my first comment (#18152 (comment))
I apologize, @cznic. The current proposal process biases bug titles to be action-based (do X) versus focused on problem statement, so I should have described the problem instead of a proposed plan of action. Maybe this is something we should consider improving in the proposal process.
The problem I see is that the ownership/maintenance story of x/net/websocket is unclear. It may suit your needs, and that is fine! However, it's not getting the same level of support as the standard library or even gorilla/websocket; it has no established owner AFAIK.
The other issue I see is along the lines of #17244, where there is concern that it elevates the status and visibility of the package. I've anecdotally heard of it confusing many new users whose needs were much better filled with gorilla/websocket. I have not heard of someone using gorilla/websocket and then deciding to use x/net/websocket instead.
And to be clear, @cznic, I'm quite serious about you maintaining x/net/websocket. As a user of it, you're better suited to be a maintainer of it than a random contributor looking for work to do. And you maintain enough other stuff that you know what the process entails.
It's actually just noise in this thread.
I respectfully disagree. I see no proof why it's different (wrt people happier with something else). I'm ready to acknowledge you're right if you can actually explain your judgement in some other way than simply claiming my point has no point (just noise in your words).
Nonetheless...
If you want to be productive, though, step up and maintain golang.org/x/net/websocket.
And to be clear, @cznic, I'm quite serious about you maintaining x/net/websocket. As a user of it, you're better suited to be a maintainer of it than a random contributor looking for work to do. And you maintain enough other stuff that you know what the process entails.
Regardless of us (at work) never having a problem with it, how do you expect me to maintain x/net/websocket? Serious question from a person having only R/O access to it.
I apologize, @cznic. ...
@zombiezen Relly appreciated, but there's nothing to apologize for. I don't agree with the proposal, but that's nothing personal. If I made an impression otherwise then it's my duty to apologize for my fault - sorry.
PS: @ALL Getting the company's approval to use Go and/or code from golang.org/whatever is one thing. The same for code from other domains is not the same.
Regardless of us (at work) never having a problem with it, how do you expect me to maintain x/net/websocket? Serious question from a person having only R/O access to it.
If you send us CLs, we'll +2 and you can submit.
If others send you CLs, we'll give you access to +2 their CLs and you can submit.
I respectfully disagree. I see no proof why it's different (wrt people happier with something else). I'm ready to acknowledge you're right if you can actually explain your judgement in some other way than simply claiming my point has no point (just noise in your words).
I find your comparison between recommending a maintained package and recommending a whole alternate language not even remotely equivalent. They're pretty absurdly different. You also found it ridiculous, which is why you prefaced it with "I really really really don't want to sound snarky". You can't just say something outlandish and preface it with "Not to be $X or anything, but ....".
By the time most people are selecting a websocket package, they've already selected their programming language (Go, in this case). On the off chance they haven't yet selected their programming language (say, they're writing a prototype application to decide whether they like the Go ecosystem), that argues even more in favor of @zombiezen's point that we should point them to the solution that's going to be make them happy, rather than the first thing they stumbled into.
Thank you Brad,
Not to add noise to the issue, but we've recently had to introduce websockets to our app and had this debate do research to find out wether x/net/websocket would work. Luckily I was aware of discussion in #17244 which mentions websocket vs the gorilla package. The mention will help others save time and avoid using something just because it looks official.
If others send you CLs, we'll give you access to +2 their CLs and you can submit.
Let's roll.
But how about implement it in standard package net/websocket
?
@leaxoy, no thanks. Unmaintained code in x/ is bad enough. We don't also want more code in std, especially if it's unmaintained. See https://golang.org/doc/faq#x_in_std
FWIW, the message in this package's godoc is not strong enough, one of my students ended up using it. You may want to consider a statement to the effect of "do not use this package, we'll murder a puppy if you do", or something. Please mention clearly what is the recommended replacement.
@jech, your students are allowed to make decisions and explore. They're students, after all.
but it still feels core enough that we should have an official answer for websockets in x/net.
My offer to donate the gorilla/websocket code to the Go project is still open.
All of the authors had signed the CLA at the time I made the offer. I don't know the status now. We will need to investigate, but let's get a preliminary decision on your end before doing the investigation.
@garyburd, a few of us chatted about it. We're fine to proceed with merging them together if you are and you can verify CLAs. (Let me know if I can help with CLA lookups). I'd like to keep the x/net package's Read and Write methods, to not break people's code, and there's a few things I'd omit or change in gorilla before adding it to x/net. (Or maybe we add it all, but mark some as Deprecated for a few months) So I'd like to do one CL that's an API review before we look at implementation code.
Then maybe we need to write a go fix
-ish tool for people.
Thoughts?
Is this the issue where we should discuss this, or should I open a new issue?
I'll follow up on email regarding licensing/CLA.
What would you omit from Gorilla package?
The move to a new import path provides an opportunity to improve the Gorilla API. The most important change is to restructure the API to allow for concurrent writers. I'd also remove the JSON methods from the connection type.
I was expecting that you would take a gorilla/websocket as a new package (x/net/websocket2 ?) or as a new types in x/net/websocket. Either way, the existing x/net/websocket would be rewritten as a wrapper around the gorilla code. The wrapper will make a best effort to be compatible the current x/net/websocket code given that x/net/websocket API exposes frames in the API while the Gorilla API exposes messages.
What would you omit from Gorilla package?
Could you send a dummy CL (to say, x/net/review/websocket) with just the API surface and all the implementation removed, and I can comment in Gerrit?
there's a few things I'd omit or change in gorilla before adding it to x/net.
Is this the right place to discuss this? If it is, then I'd like to mention that we're using Websockets for sending frequent tiny updates (on the order of a few octets), and that a cursory reading through the Gorilla implementation doesn't show a way to send a small data frame without consing up a writer.
@jech, that sounds like an implementation detail, but it important to keep in mind since the choice of API does have an impact on performance.
@jech See gorilla/websocket@0868951. The message writer is no longer allocated on heap for the WriteMessage fast path.
any updates?
There's agreement about moving the package, but I don't have time to do the work.
Google Cloud Platform has a beta websocket API: https://issuetracker.google.com/issues/35886348
Despite lack of maintainers I'd still like to reopen the idea of a standard library package net/websocket for Go 2.
I think I may be encountering this error when using Gorilla's websockets, which apparently isn't present in this implementation: gorilla/websocket#43 .
I have a WebSocket server, with two goroutines per connection: one reading messages and pushing them to a channel, and the other pulling them from the channel, processing them and writing the results to the connection. So far, it looks like at semi-random points I get spurious "unexpected EOF" messages on the Gorilla implementation. I yet need to try Google's.
Looks like Gorilla's ws implementation needs a maintainer.
@garyburd regarding your comment above
The most important change is to restructure the API to allow for concurrent writers.
Why is that important and what would this sort of API look like?
I started working on a new WebSocket library for Go.
The main features being a minimal easy to use API that will work with WebSockets over HTTP/2.
https://github.com/nhooyr/websocket#websocket
Right now the entire implementation is stubbed out as I wanted to get some feedback on the API.
I just tagged v0.1.0, it passes all of the autobahn tests except for compression which I'll add later.
https://github.com/nhooyr/websocket
Let me know what you think.
edit: Have tagged v1.0.0 now 🎊
I opened a PR to add my library to the deprecation message as well. golang/net#51
Should we make the deprecation message stronger as discussed in #33215?
It seems to actively have issues with pings as described by @fjl at ethereum/go-ethereum#19798 (comment)
The issue here isn't that github.com/x/net/websocket doesn't handle ping (it does). We are seeing the reproducer behave this way because SetWriteDeadline interacts badly with the built-in ping frame handler. Whenever the RPC client wants to write something, it sets a write deadline. In the reproducer it only writes once to establish the subscription. When the ping frame arrives 12 seconds after the write, the write deadline that's set has already passed and writing the pong frame fails. The pong write error is returned from Read, causing the client to reconnect.
I looked over the code and confirmed this is an issue. The ping handling code never resets the write deadline and instead relies on the user of the Conn to reset the deadline to zero after reading which is definitely confusing behaviour. I suppose this could be fixed pretty easily but people ought to just use gorilla or mine. Mine also has a net.Conn wrapper to make such a transition easier. I'll add this to the deprecation message.
try https://github.com/lxzan/gws , provide better performance, easier to use API
这个包满足我的,因为很小,而且是官方维护。唯一不足的地方大多数人没法用它,因为不会Read。
var c *websocket.Conn
var buf bytes.Buffer
b := make([]byte, 1024)
for {
n, err := c.Read(b)
if n > 0 {
buf.Write(b[:n])
} else if err != nil {
return
}
//处理帧数据,你可能有如下三种判断之一:
if err == io.EOF { //表示帧结束? 但是你可能永远等不到io.EOF这个值
//处理帧数据
buf.Reset()
}
//或者
if buf.Len() == c.Len() { //c.Len()? 这方法返回没有准确表达我需要的条件,不知道有什么作用:frame.Length?
//处理帧数据
buf.Reset()
}
//或者
if buf.Len() == int(c.FrameDataLenth()) { //ok,非常棒。但是 websocket.Conn 没有这个方法:FrameDataLenth=frame.header.Length
//处理帧数据
buf.Reset()
}
}
@bradfitz do you think it's possible to continue the conversation about merging gorilla/webosocket into x/net/websocket now that they have new maintainers or that ship has sailed a long time ago?
Hello, is there any update on this? It's been a long time since gorilla/ws got maintainers.
I really want to have good ws solution in stdlib.
Hello, is there any update on this? It's been a long time since gorilla/ws got maintainers.
I really want to have good ws solution in stdlib.
'x/' packages like x/net/websocket aren't part of the standard library either. If you like gorilla/websocket just use it.
Additionally, in light of the recent xz open source attack, we've all been reminded that sending pressure comments to open source maintainers, urging them to incorporate code from individuals they don't personally know, is not advisable. The code of gorilla/websocket wasn't created with the same review standards as packages by the Go project. And I personally feel that it has gotten worse after the change in maintainership, see gorilla/websocket#873
Change https://go.dev/cl/614075 mentions this issue: websocket: update nhooyr.io/websocket to github.com/coder/websocket
I am trying to go a:
go get -v golang.org/x/net/websocket
and getting
Fetching https://golang.org/x/net/websocket?go-get=1
Parsing meta tags from https://golang.org/x/net/websocket?go-get=1 (status code 200)
import "golang.org/x/net/websocket": parsing golang.org/x/net/websocket: XML syntax error on line 7: unexpected end element
package golang.org/x/net/websocket: unrecognized import path "golang.org/x/net/websocket"