2019-06-02 16:49:45 +02:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"log"
|
|
|
|
|
|
|
|
"github.com/kataras/iris"
|
|
|
|
"github.com/kataras/iris/websocket"
|
2019-07-01 17:52:17 +02:00
|
|
|
|
2019-07-06 05:27:28 +02:00
|
|
|
// Used when "enableJWT" constant is true:
|
|
|
|
"github.com/dgrijalva/jwt-go"
|
|
|
|
jwtmiddleware "github.com/iris-contrib/middleware/jwt"
|
2019-06-02 16:49:45 +02:00
|
|
|
)
|
|
|
|
|
2019-07-06 05:27:28 +02:00
|
|
|
// values should match with the client sides as well.
|
|
|
|
const enableJWT = true
|
2019-06-02 16:49:45 +02:00
|
|
|
const namespace = "default"
|
|
|
|
|
2019-07-11 11:59:11 +02:00
|
|
|
// if namespace is empty then simply websocket.Events{...} can be used instead.
|
|
|
|
var serverEvents = websocket.Namespaces{
|
|
|
|
namespace: websocket.Events{
|
|
|
|
websocket.OnNamespaceConnected: func(nsConn *websocket.NSConn, msg websocket.Message) error {
|
2019-07-01 17:52:17 +02:00
|
|
|
// with `websocket.GetContext` you can retrieve the Iris' `Context`.
|
|
|
|
ctx := websocket.GetContext(nsConn.Conn)
|
|
|
|
|
2019-06-02 17:27:51 +02:00
|
|
|
log.Printf("[%s] connected to namespace [%s] with IP [%s]",
|
|
|
|
nsConn, msg.Namespace,
|
2019-07-01 17:52:17 +02:00
|
|
|
ctx.RemoteAddr())
|
2019-06-02 16:49:45 +02:00
|
|
|
return nil
|
|
|
|
},
|
2019-07-11 11:59:11 +02:00
|
|
|
websocket.OnNamespaceDisconnect: func(nsConn *websocket.NSConn, msg websocket.Message) error {
|
2019-06-02 17:27:51 +02:00
|
|
|
log.Printf("[%s] disconnected from namespace [%s]", nsConn, msg.Namespace)
|
2019-06-02 16:49:45 +02:00
|
|
|
return nil
|
|
|
|
},
|
2019-07-11 11:59:11 +02:00
|
|
|
"chat": func(nsConn *websocket.NSConn, msg websocket.Message) error {
|
2019-06-02 17:27:51 +02:00
|
|
|
// room.String() returns -> NSConn.String() returns -> Conn.String() returns -> Conn.ID()
|
|
|
|
log.Printf("[%s] sent: %s", nsConn, string(msg.Body))
|
2019-06-02 16:49:45 +02:00
|
|
|
|
|
|
|
// Write message back to the client message owner with:
|
2019-06-02 17:27:51 +02:00
|
|
|
// nsConn.Emit("chat", msg)
|
2019-06-02 16:49:45 +02:00
|
|
|
// Write message to all except this client with:
|
2019-06-02 17:27:51 +02:00
|
|
|
nsConn.Conn.Server().Broadcast(nsConn, msg)
|
2019-06-02 16:49:45 +02:00
|
|
|
return nil
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
app := iris.New()
|
2019-07-11 11:59:11 +02:00
|
|
|
websocketServer := websocket.New(
|
2019-06-13 11:15:01 +02:00
|
|
|
websocket.DefaultGorillaUpgrader, /* DefaultGobwasUpgrader can be used too. */
|
2019-06-02 16:49:45 +02:00
|
|
|
serverEvents)
|
|
|
|
|
2019-07-06 05:27:28 +02:00
|
|
|
j := jwtmiddleware.New(jwtmiddleware.Config{
|
|
|
|
// Extract by the "token" url,
|
|
|
|
// so the client should dial with ws://localhost:8080/echo?token=$token
|
|
|
|
Extractor: jwtmiddleware.FromParameter("token"),
|
|
|
|
|
|
|
|
ValidationKeyGetter: func(token *jwt.Token) (interface{}, error) {
|
|
|
|
return []byte("My Secret"), nil
|
|
|
|
},
|
|
|
|
// When set, the middleware verifies that tokens are signed with the specific signing algorithm
|
|
|
|
// If the signing method is not constant the ValidationKeyGetter callback can be used to implement additional checks
|
|
|
|
// Important to avoid security issues described here: https://auth0.com/blog/2015/03/31/critical-vulnerabilities-in-json-web-token-libraries/
|
|
|
|
SigningMethod: jwt.SigningMethodHS256,
|
|
|
|
})
|
|
|
|
|
2019-06-02 16:49:45 +02:00
|
|
|
// serves the endpoint of ws://localhost:8080/echo
|
2019-07-06 05:27:28 +02:00
|
|
|
websocketRoute := app.Get("/echo", websocket.Handler(websocketServer))
|
2019-07-09 11:25:06 +02:00
|
|
|
|
2019-07-06 05:27:28 +02:00
|
|
|
if enableJWT {
|
|
|
|
// Register the jwt middleware (on handshake):
|
|
|
|
websocketRoute.Use(j.Serve)
|
|
|
|
// OR
|
|
|
|
//
|
|
|
|
// Check for token through the jwt middleware
|
|
|
|
// on websocket connection or on any event:
|
2019-07-11 11:59:11 +02:00
|
|
|
/* websocketServer.OnConnect = func(c *websocket.Conn) error {
|
2019-07-09 11:16:19 +02:00
|
|
|
ctx := websocket.GetContext(c)
|
|
|
|
if err := j.CheckJWT(ctx); err != nil {
|
|
|
|
// will send the above error on the client
|
|
|
|
// and will not allow it to connect to the websocket server at all.
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
user := ctx.Values().Get("jwt").(*jwt.Token)
|
|
|
|
// or just: user := j.Get(ctx)
|
|
|
|
|
|
|
|
log.Printf("This is an authenticated request\n")
|
|
|
|
log.Printf("Claim content:")
|
|
|
|
log.Printf("%#+v\n", user.Claims)
|
|
|
|
|
|
|
|
log.Printf("[%s] connected to the server", c.ID())
|
|
|
|
|
|
|
|
return nil
|
|
|
|
} */
|
2019-07-06 05:27:28 +02:00
|
|
|
}
|
2019-06-02 16:49:45 +02:00
|
|
|
|
|
|
|
// serves the browser-based websocket client.
|
|
|
|
app.Get("/", func(ctx iris.Context) {
|
|
|
|
ctx.ServeFile("./browser/index.html", false)
|
|
|
|
})
|
|
|
|
|
|
|
|
// serves the npm browser websocket client usage example.
|
2019-06-21 18:43:25 +02:00
|
|
|
app.HandleDir("/browserify", "./browserify")
|
2019-06-02 16:49:45 +02:00
|
|
|
|
2019-06-13 11:15:01 +02:00
|
|
|
app.Run(iris.Addr(":8080"), iris.WithoutServerError(iris.ErrServerClosed))
|
2019-06-02 16:49:45 +02:00
|
|
|
}
|