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

wspb: Unexpected removal of publicly used module broke everything! #420

Closed
xornet-sl opened this issue Nov 12, 2023 · 3 comments
Closed

Comments

@xornet-sl
Copy link

refers to #311

Is this a joke? Why wspb - a public package got removed?!
Guys, you exposed this, people use it, it is NOT an internal package. Why you just brake everything in PATCH release??!!
I even can not re-create the functionality of wspb as it uses internal package. Why you did that? Was it really a problem just to update the needed dependencies and let it live?

I was updating websocket from 1.8.7 to 1.8.10 and it broke the whole application. It's not a good approach to introduce any breaking changes in PATCH revisions. Mark it deprecated or fix it. Please, perform removals like that in MAJOR updates.
Now I have to fork your repo and maintain it separately or stick with older version. Or most probably find an alternative.

@nhooyr
Copy link
Contributor

nhooyr commented Nov 12, 2023

It never belonged in the library and is very easy to recreate. The only reason for internal being imported is for buffer repooling which isn't that necessary but also not much code if you want it. See https://github.com/nhooyr/websocket/blob/docs/internal/bpool/bpool.go

It is unfortunate to remove in a patch release I'll agree. I didn't realize they reimplemented github.com/golang/protobuf in terms of google.golang.org/protobuf so there's really nothing wrong with using it right now but regardless, I wanted to make go.mod and go.sum empty, with zero transitive dependencies and I didn't want to wait for v2. See also #297

Here's a standalone implementation:

// Package wspb provides helpers for reading and writing protobuf messages.
package wspb

import (
	"bytes"
	"context"
	"fmt"
	"sync"

	"github.com/golang/protobuf/proto"
	"nhooyr.io/websocket"
)

// Read reads a protobuf message from c into v.
// It will reuse buffers in between calls to avoid allocations.
func Read(ctx context.Context, c *websocket.Conn, v proto.Message) error {
	return read(ctx, c, v)
}

func read(ctx context.Context, c *websocket.Conn, v proto.Message) (err error) {
	defer errdWrap(&err, "failed to read protobuf message")

	typ, r, err := c.Reader(ctx)
	if err != nil {
		return err
	}

	if typ != websocket.MessageBinary {
		c.Close(websocket.StatusUnsupportedData, "expected binary message")
		return fmt.Errorf("expected binary message for protobuf but got: %v", typ)
	}

	b := bpoolGet()
	defer bpoolPut(b)

	_, err = b.ReadFrom(r)
	if err != nil {
		return err
	}

	err = proto.Unmarshal(b.Bytes(), v)
	if err != nil {
		c.Close(websocket.StatusInvalidFramePayloadData, "failed to unmarshal protobuf")
		return fmt.Errorf("failed to unmarshal protobuf: %w", err)
	}

	return nil
}

// Write writes the protobuf message v to c.
// It will reuse buffers in between calls to avoid allocations.
func Write(ctx context.Context, c *websocket.Conn, v proto.Message) error {
	return write(ctx, c, v)
}

func write(ctx context.Context, c *websocket.Conn, v proto.Message) (err error) {
	defer errdWrap(&err, "failed to write protobuf message")

	b := bpoolGet()
	pb := proto.NewBuffer(b.Bytes())
	defer func() {
		bpoolPut(bytes.NewBuffer(pb.Bytes()))
	}()

	err = pb.Marshal(v)
	if err != nil {
		return fmt.Errorf("failed to marshal protobuf: %w", err)
	}

	return c.Write(ctx, websocket.MessageBinary, pb.Bytes())
}

// errdWrap wraps err with fmt.Errorf if err is non nil.
// Intended for use with defer and a named error return.
// Inspired by https://github.com/golang/go/issues/32676.
func errdWrap(err *error, f string, v ...interface{}) {
	if *err != nil {
		*err = fmt.Errorf(f+": %w", append(v, *err)...)
	}
}

var bpool sync.Pool

// Get returns a buffer from the pool or creates a new one if
// the pool is empty.
func bpoolGet() *bytes.Buffer {
	b := bpool.Get()
	if b == nil {
		return &bytes.Buffer{}
	}
	return b.(*bytes.Buffer)
}

// Put returns a buffer into the pool.
func bpoolPut(b *bytes.Buffer) {
	b.Reset()
	bpool.Put(b)
}

@nhooyr nhooyr closed this as completed Nov 12, 2023
@xornet-sl
Copy link
Author

Thanks for the fast reply, appreciate it. I'll try to use your suggested implementation.

It never belonged in the library

How come? It was in the package, it could be imported and it uses library's code. I understand that it was an utility module, but it definitely belonged to the package and was very useful along with websocket itself. Just like, for example wsjon, which hasn't (yet?) been deleted.

@nhooyr
Copy link
Contributor

nhooyr commented Nov 12, 2023

How come? It was in the package, it could be imported and it uses library's code. I understand that it was an utility module, but it definitely belonged to the package and was very useful along with websocket itself. Just like, for example wsjon, which hasn't (yet?) been deleted.

I agree it was useful. The only reason I decided to remove it is because of the third party dependency. wsjson uses the standard library encoding/json implementation and so doesn't add any bloat to the library. It irked a lot of people that the library didn't have an empty go.sum and go.mod as compared to Gorilla.

xornet-sl added a commit to xornet-sl/go-xrpc that referenced this issue Feb 21, 2024
due to its removal from nhooyr/websocket:
coder/websocket#420
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants