websocket actions: Refactor registration
This commit is contained in:
parent
08ab7b4383
commit
7b5f8a9882
4 changed files with 102 additions and 87 deletions
|
|
@ -3,42 +3,80 @@ package socket
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
|
"github.com/FAU-CDI/wisski-distillery/internal/dis/component/server/admin/socket/actions"
|
||||||
|
"github.com/FAU-CDI/wisski-distillery/internal/dis/component/server/admin/socket/proto"
|
||||||
|
"github.com/rs/zerolog"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (sockets *Sockets) Actions() ActionMap {
|
func (sockets *Sockets) Actions(ctx context.Context) proto.ActionMap {
|
||||||
actions := make(ActionMap, len(sockets.dependencies.Actions)+len(sockets.dependencies.IActions))
|
logger := zerolog.Ctx(ctx)
|
||||||
|
actions := make(proto.ActionMap, len(sockets.dependencies.Actions)+len(sockets.dependencies.IActions))
|
||||||
|
|
||||||
// setup basic actions
|
// setup basic actions
|
||||||
for _, a := range sockets.dependencies.Actions {
|
for _, a := range sockets.dependencies.Actions {
|
||||||
a := a
|
name, action := sockets.regularAction(a)
|
||||||
meta := a.Action()
|
if _, ok := actions[name]; ok {
|
||||||
actions[meta.Name] = Action{
|
logger.Warn().Str("name", name).Str("type", "regular").Msg("duplicate websocket action")
|
||||||
NumParams: meta.NumParams,
|
|
||||||
Scope: meta.Scope,
|
|
||||||
ScopeParam: meta.ScopeParam,
|
|
||||||
|
|
||||||
Handle: a.Act,
|
|
||||||
}
|
}
|
||||||
|
actions[name] = action
|
||||||
|
|
||||||
|
logger.Info().
|
||||||
|
Str("name", name).
|
||||||
|
Str("type", "regular").
|
||||||
|
Int("params", action.NumParams).
|
||||||
|
Str("scope", string(action.Scope)).
|
||||||
|
Str("scopeParam", action.ScopeParam).
|
||||||
|
Msg("registering websocket action")
|
||||||
}
|
}
|
||||||
|
|
||||||
// setup instance actions
|
// setup instance actions
|
||||||
for _, a := range sockets.dependencies.IActions {
|
for _, a := range sockets.dependencies.IActions {
|
||||||
a := a
|
name, action := sockets.instanceAction(a)
|
||||||
meta := a.Action()
|
if _, ok := actions[name]; ok {
|
||||||
actions[meta.Name] = Action{
|
zerolog.Ctx(ctx).Warn().Str("name", name).Str("type", "instance").Msg("duplicate websocket action")
|
||||||
NumParams: meta.NumParams + 1,
|
|
||||||
Scope: meta.Scope,
|
|
||||||
ScopeParam: meta.ScopeParam,
|
|
||||||
|
|
||||||
Handle: func(ctx context.Context, in io.Reader, out io.Writer, params ...string) error {
|
|
||||||
instance, err := sockets.dependencies.Instances.WissKI(ctx, params[0])
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return a.Act(ctx, instance, in, out, params[1:]...)
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
|
actions[name] = action
|
||||||
|
|
||||||
|
logger.Info().
|
||||||
|
Str("name", name).
|
||||||
|
Str("type", "instance").
|
||||||
|
Int("params", action.NumParams-1).
|
||||||
|
Str("scope", string(action.Scope)).
|
||||||
|
Str("scopeParam", action.ScopeParam).
|
||||||
|
Msg("registering websocket action")
|
||||||
}
|
}
|
||||||
|
|
||||||
return actions
|
return actions
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (sockets *Sockets) regularAction(a actions.WebsocketAction) (name string, action proto.Action) {
|
||||||
|
meta := a.Action()
|
||||||
|
|
||||||
|
return meta.Name, proto.Action{
|
||||||
|
NumParams: meta.NumParams + 1,
|
||||||
|
Scope: meta.Scope,
|
||||||
|
ScopeParam: meta.ScopeParam,
|
||||||
|
|
||||||
|
Handle: a.Act,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (sockets *Sockets) instanceAction(a actions.WebsocketInstanceAction) (name string, action proto.Action) {
|
||||||
|
meta := a.Action()
|
||||||
|
|
||||||
|
return meta.Name, proto.Action{
|
||||||
|
NumParams: meta.NumParams + 1,
|
||||||
|
Scope: meta.Scope,
|
||||||
|
ScopeParam: meta.ScopeParam,
|
||||||
|
|
||||||
|
Handle: func(ctx context.Context, in io.Reader, out io.Writer, params ...string) error {
|
||||||
|
instance, err := sockets.dependencies.Instances.WissKI(ctx, params[0])
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return a.Act(ctx, instance, in, out, params[1:]...)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
|
||||||
36
internal/dis/component/server/admin/socket/proto/info.go
Normal file
36
internal/dis/component/server/admin/socket/proto/info.go
Normal file
|
|
@ -0,0 +1,36 @@
|
||||||
|
package proto
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"io"
|
||||||
|
|
||||||
|
"github.com/FAU-CDI/wisski-distillery/internal/dis/component"
|
||||||
|
"github.com/FAU-CDI/wisski-distillery/internal/dis/component/auth/scopes"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Action is anything that can be retrieved from a
|
||||||
|
type Action struct {
|
||||||
|
// NumPara
|
||||||
|
NumParams int
|
||||||
|
|
||||||
|
// Scope and ScopeParam indicate the scope required by the caller.
|
||||||
|
// TODO(twiesing): Once we actually include scopes, make them dynamic
|
||||||
|
Scope component.Scope
|
||||||
|
ScopeParam string
|
||||||
|
|
||||||
|
// Handle handles this action.
|
||||||
|
//
|
||||||
|
// ctx is closed once the underlying connection is closed.
|
||||||
|
// out is an io.Writer that is automatically sent to the client.
|
||||||
|
// params holds exactly NumParams parameters.
|
||||||
|
Handle func(ctx context.Context, in io.Reader, out io.Writer, params ...string) error
|
||||||
|
}
|
||||||
|
|
||||||
|
// scope returns the actual scope required by this action.
|
||||||
|
// If the caller did not provide an actual scope, uses ScopeNever
|
||||||
|
func (action Action) scope() component.Scope {
|
||||||
|
if action.Scope == "" {
|
||||||
|
return scopes.ScopeNever
|
||||||
|
}
|
||||||
|
return action.Scope
|
||||||
|
}
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
package socket
|
package proto
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
|
@ -9,9 +9,7 @@ import (
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/FAU-CDI/wisski-distillery/internal/dis/component"
|
|
||||||
"github.com/FAU-CDI/wisski-distillery/internal/dis/component/auth"
|
"github.com/FAU-CDI/wisski-distillery/internal/dis/component/auth"
|
||||||
"github.com/FAU-CDI/wisski-distillery/internal/dis/component/auth/scopes"
|
|
||||||
"github.com/gorilla/websocket"
|
"github.com/gorilla/websocket"
|
||||||
"github.com/tkw1536/pkglib/httpx"
|
"github.com/tkw1536/pkglib/httpx"
|
||||||
)
|
)
|
||||||
|
|
@ -195,33 +193,6 @@ func (wf WriteFunc) Write(b []byte) (int, error) {
|
||||||
return wf(b)
|
return wf(b)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Action is something that can be handled via a WebSocket connection.
|
|
||||||
type Action struct {
|
|
||||||
// NumPara
|
|
||||||
NumParams int
|
|
||||||
|
|
||||||
// Scope and ScopeParam indicate the scope required by the caller.
|
|
||||||
// TODO(twiesing): Once we actually include scopes, make them dynamic
|
|
||||||
Scope component.Scope
|
|
||||||
ScopeParam string
|
|
||||||
|
|
||||||
// Handle handles this action.
|
|
||||||
//
|
|
||||||
// ctx is closed once the underlying connection is closed.
|
|
||||||
// out is an io.Writer that is automatically sent to the client.
|
|
||||||
// params holds exactly NumParams parameters.
|
|
||||||
Handle func(ctx context.Context, in io.Reader, out io.Writer, params ...string) error
|
|
||||||
}
|
|
||||||
|
|
||||||
// scope returns the actual scope required by this action.
|
|
||||||
// If the caller did not provide an actual scope, uses ScopeNever
|
|
||||||
func (action Action) scope() component.Scope {
|
|
||||||
if action.Scope == "" {
|
|
||||||
return scopes.ScopeNever
|
|
||||||
}
|
|
||||||
return action.Scope
|
|
||||||
}
|
|
||||||
|
|
||||||
// CallMessage is sent by the client to the server to invoke a remote procedure
|
// CallMessage is sent by the client to the server to invoke a remote procedure
|
||||||
type CallMessage struct {
|
type CallMessage struct {
|
||||||
Call string `json:"call"`
|
Call string `json:"call"`
|
||||||
|
|
@ -2,7 +2,6 @@ package socket
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"io"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/FAU-CDI/wisski-distillery/internal/dis/component"
|
"github.com/FAU-CDI/wisski-distillery/internal/dis/component"
|
||||||
|
|
@ -13,7 +12,7 @@ import (
|
||||||
"github.com/FAU-CDI/wisski-distillery/internal/dis/component/instances/purger"
|
"github.com/FAU-CDI/wisski-distillery/internal/dis/component/instances/purger"
|
||||||
"github.com/FAU-CDI/wisski-distillery/internal/dis/component/provision"
|
"github.com/FAU-CDI/wisski-distillery/internal/dis/component/provision"
|
||||||
"github.com/FAU-CDI/wisski-distillery/internal/dis/component/server/admin/socket/actions"
|
"github.com/FAU-CDI/wisski-distillery/internal/dis/component/server/admin/socket/actions"
|
||||||
"github.com/FAU-CDI/wisski-distillery/internal/wisski"
|
"github.com/FAU-CDI/wisski-distillery/internal/dis/component/server/admin/socket/proto"
|
||||||
"github.com/rs/zerolog"
|
"github.com/rs/zerolog"
|
||||||
"github.com/tkw1536/pkglib/httpx"
|
"github.com/tkw1536/pkglib/httpx"
|
||||||
"github.com/tkw1536/pkglib/lazy"
|
"github.com/tkw1536/pkglib/lazy"
|
||||||
|
|
@ -22,7 +21,7 @@ import (
|
||||||
type Sockets struct {
|
type Sockets struct {
|
||||||
component.Base
|
component.Base
|
||||||
|
|
||||||
actions lazy.Lazy[ActionMap]
|
actions lazy.Lazy[proto.ActionMap]
|
||||||
|
|
||||||
dependencies struct {
|
dependencies struct {
|
||||||
Actions []actions.WebsocketAction
|
Actions []actions.WebsocketAction
|
||||||
|
|
@ -58,37 +57,8 @@ func (sockets *Sockets) HandleRoute(ctx context.Context, path string) (http.Hand
|
||||||
// Serve handles a connection to the websocket api
|
// Serve handles a connection to the websocket api
|
||||||
func (socket *Sockets) Serve(conn httpx.WebSocketConnection) {
|
func (socket *Sockets) Serve(conn httpx.WebSocketConnection) {
|
||||||
// handle the websocket connection!
|
// handle the websocket connection!
|
||||||
name, err := socket.actions.Get(socket.Actions).Handle(socket.dependencies.Auth, conn)
|
name, err := socket.actions.Get(func() proto.ActionMap { return socket.Actions(conn.Context()) }).Handle(socket.dependencies.Auth, conn)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
zerolog.Ctx(conn.Context()).Err(err).Str("name", name).Msg("Error handling websocket")
|
zerolog.Ctx(conn.Context()).Err(err).Str("name", name).Msg("Error handling websocket")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Generic returns a new action that calls handler with the provided number of parameters
|
|
||||||
func (sockets *Sockets) Generic(scope component.Scope, scopeParam string, numParams int, handler func(ctx context.Context, socket *Sockets, in io.Reader, out io.Writer, params ...string) error) Action {
|
|
||||||
return Action{
|
|
||||||
Scope: scope,
|
|
||||||
ScopeParam: scopeParam,
|
|
||||||
NumParams: numParams,
|
|
||||||
Handle: func(ctx context.Context, in io.Reader, out io.Writer, params ...string) error {
|
|
||||||
return handler(ctx, sockets, in, out, params...)
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Insstance returns a new action that calls handler with a specific WissKI instance
|
|
||||||
func (sockets *Sockets) Instance(scope component.Scope, scopeParam string, numParams int, handler func(ctx context.Context, sockets *Sockets, instance *wisski.WissKI, in io.Reader, out io.Writer, params ...string) error) Action {
|
|
||||||
return Action{
|
|
||||||
Scope: scope,
|
|
||||||
ScopeParam: scopeParam,
|
|
||||||
|
|
||||||
NumParams: numParams + 1,
|
|
||||||
Handle: func(ctx context.Context, in io.Reader, out io.Writer, params ...string) error {
|
|
||||||
instance, err := sockets.dependencies.Instances.WissKI(ctx, params[0])
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return handler(ctx, sockets, instance, in, out, params[1:]...)
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue