send and receive messages
parent
dd8762667d
commit
201ebfe8b5
@ -0,0 +1,13 @@
|
||||
package main
|
||||
|
||||
func errorResponse(re int, err error) response {
|
||||
var body struct {
|
||||
Message string `json:"message"`
|
||||
}
|
||||
body.Message = err.Error()
|
||||
return response{
|
||||
Re: re,
|
||||
Type: "error",
|
||||
Body: body,
|
||||
}
|
||||
}
|
@ -0,0 +1,38 @@
|
||||
package errors
|
||||
|
||||
import (
|
||||
std "errors"
|
||||
)
|
||||
|
||||
// New returns an error that formats as the given text.
|
||||
// Each call to New returns a distinct error value even if the text is identical.
|
||||
func New(text string) error { return std.New(text) }
|
||||
|
||||
// Unwrap returns the result of calling the Unwrap method on err, if err's
|
||||
// type contains an Unwrap method returning error.
|
||||
// Otherwise, Unwrap returns nil.
|
||||
func Unwrap(err error) error { return std.Unwrap(err) }
|
||||
|
||||
// Is reports whether any error in err's chain matches target.
|
||||
//
|
||||
// The chain consists of err itself followed by the sequence of errors obtained by
|
||||
// repeatedly calling Unwrap.
|
||||
//
|
||||
// An error is considered to match a target if it is equal to that target or if
|
||||
// it implements a method Is(error) bool such that Is(target) returns true.
|
||||
func Is(err, target error) bool { return std.Is(err, target) }
|
||||
|
||||
// As finds the first error in err's chain that matches target, and if so, sets
|
||||
// target to that error value and returns true.
|
||||
//
|
||||
// The chain consists of err itself followed by the sequence of errors obtained by
|
||||
// repeatedly calling Unwrap.
|
||||
//
|
||||
// An error matches target if the error's concrete value is assignable to the value
|
||||
// pointed to by target, or if the error has a method As(interface{}) bool such that
|
||||
// As(target) returns true. In the latter case, the As method is responsible for
|
||||
// setting target.
|
||||
//
|
||||
// As will panic if target is not a non-nil pointer to either a type that implements
|
||||
// error, or to any interface type. As returns false if err is nil.
|
||||
func As(err error, target interface{}) bool { return std.As(err, target) }
|
@ -0,0 +1,14 @@
|
||||
package main
|
||||
|
||||
type response struct {
|
||||
Re int `json:"re"`
|
||||
Type string `json:"type"`
|
||||
Body interface{} `json:"body,omitempty"`
|
||||
}
|
||||
|
||||
func ok(re int) response {
|
||||
return response{
|
||||
Re: re,
|
||||
Type: "ok",
|
||||
}
|
||||
}
|
@ -0,0 +1,56 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"context"
|
||||
"encoding/json"
|
||||
"time"
|
||||
|
||||
"github.com/gorilla/websocket"
|
||||
"github.com/jordanorelli/blammo"
|
||||
)
|
||||
|
||||
// session represents the server side of a client's session. i.e., a single
|
||||
// connection along with its associated state.
|
||||
type session struct {
|
||||
*blammo.Log
|
||||
id int
|
||||
conn *websocket.Conn
|
||||
outbox chan response
|
||||
}
|
||||
|
||||
// pump is the session send loop. Pump should pump the session's outbox
|
||||
// messages to the underlying connection until the context is closed.
|
||||
func (sn *session) pump(ctx context.Context) {
|
||||
for {
|
||||
select {
|
||||
case res := <-sn.outbox:
|
||||
payload, err := json.Marshal(res)
|
||||
if err != nil {
|
||||
sn.Error("failed to marshal outgoing response: %v", err)
|
||||
break
|
||||
}
|
||||
|
||||
if err := sn.conn.SetWriteDeadline(time.Now().Add(3 * time.Second)); err != nil {
|
||||
sn.Error("failed to set write deadline: %v", err)
|
||||
break
|
||||
}
|
||||
w, err := sn.conn.NextWriter(websocket.TextMessage)
|
||||
if err != nil {
|
||||
sn.Error("failed get a writer frame: %v", err)
|
||||
break
|
||||
}
|
||||
if _, err := w.Write(payload); err != nil {
|
||||
sn.Error("failed write payload: %v", err)
|
||||
break
|
||||
}
|
||||
if err := w.Close(); err != nil {
|
||||
sn.Error("failed to close write frame: %v", err)
|
||||
break
|
||||
}
|
||||
sn.Child("sent-frame").Info(string(payload))
|
||||
case <-ctx.Done():
|
||||
sn.Info("parent context done, shutting down write pump")
|
||||
return
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue