Add basic websocket structure and request upgrader
This commit is contained in:
@ -32,12 +32,12 @@ func provideRunner(writer http.ResponseWriter, request *http.Request) {
|
|||||||
writeNotFound(writer, err)
|
writeNotFound(writer, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
runner, err := env.NextRunner()
|
nextRunner, err := env.NextRunner()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
writeInternalServerError(writer, err, dto.ErrorNomadOverload)
|
writeInternalServerError(writer, err, dto.ErrorNomadOverload)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
sendJson(writer, &dto.RunnerResponse{Id: runner.Id()}, http.StatusOK)
|
sendJson(writer, &dto.RunnerResponse{Id: nextRunner.Id()}, http.StatusOK)
|
||||||
}
|
}
|
||||||
|
|
||||||
// executeCommand takes an ExecutionRequest and stores it for a runner.
|
// executeCommand takes an ExecutionRequest and stores it for a runner.
|
||||||
@ -80,14 +80,6 @@ func executeCommand(router *mux.Router) func(w http.ResponseWriter, r *http.Requ
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// connectToRunner is a placeholder for now and will become the endpoint for websocket connections.
|
|
||||||
func connectToRunner(writer http.ResponseWriter, request *http.Request) {
|
|
||||||
// Todo: Execute the command, upgrade the connection to websocket and handle forwarding
|
|
||||||
executionId := request.URL.Query()[ExecutionIdKey]
|
|
||||||
log.WithField("executionId", executionId).Info("Websocket for execution requested.")
|
|
||||||
writer.WriteHeader(http.StatusNotImplemented)
|
|
||||||
}
|
|
||||||
|
|
||||||
// The findRunnerMiddleware looks up the runnerId for routes containing it
|
// The findRunnerMiddleware looks up the runnerId for routes containing it
|
||||||
// and adds the runner to the context of the request.
|
// and adds the runner to the context of the request.
|
||||||
func findRunnerMiddleware(runnerPool environment.RunnerPool) func(handler http.Handler) http.Handler {
|
func findRunnerMiddleware(runnerPool environment.RunnerPool) func(handler http.Handler) http.Handler {
|
||||||
|
46
api/websocket.go
Normal file
46
api/websocket.go
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
package api
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"github.com/gorilla/websocket"
|
||||||
|
"gitlab.hpi.de/codeocean/codemoon/poseidon/api/dto"
|
||||||
|
"gitlab.hpi.de/codeocean/codemoon/poseidon/runner"
|
||||||
|
"net/http"
|
||||||
|
)
|
||||||
|
|
||||||
|
var connUpgrade = websocket.Upgrader{
|
||||||
|
ReadBufferSize: 1024,
|
||||||
|
WriteBufferSize: 1024,
|
||||||
|
}
|
||||||
|
|
||||||
|
// connectToRunner is a placeholder for now and will become the endpoint for websocket connections.
|
||||||
|
func connectToRunner(writer http.ResponseWriter, request *http.Request) {
|
||||||
|
r, ok := runner.FromContext(request.Context())
|
||||||
|
if !ok {
|
||||||
|
log.Error("Runner not set in request context.")
|
||||||
|
writeInternalServerError(writer, errors.New("findRunnerMiddleware failure"), dto.ErrorUnknown)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
executionId := request.URL.Query().Get(ExecutionIdKey)
|
||||||
|
connClient, err := connUpgrade.Upgrade(writer, request, nil)
|
||||||
|
if err != nil {
|
||||||
|
writeInternalServerError(writer, err, dto.ErrorUnknown)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer func(connClient *websocket.Conn) {
|
||||||
|
err := connClient.Close()
|
||||||
|
if err != nil {
|
||||||
|
writeInternalServerError(writer, err, dto.ErrorUnknown)
|
||||||
|
}
|
||||||
|
}(connClient)
|
||||||
|
|
||||||
|
// ToDo: Implement communication forwarding
|
||||||
|
err, ok = r.Execute(runner.ExecutionId(executionId))
|
||||||
|
if !ok {
|
||||||
|
writeBadRequest(writer, errors.New("invalid Execution Id"))
|
||||||
|
return
|
||||||
|
} else if err != nil {
|
||||||
|
writeInternalServerError(writer, err, dto.ErrorUnknown)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
12
api/websocket_test.go
Normal file
12
api/websocket_test.go
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
package api
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestInvalidExecutionId(t *testing.T) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestEstablishWebsocketConnection(t *testing.T) {
|
||||||
|
}
|
@ -5,10 +5,14 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
"gitlab.hpi.de/codeocean/codemoon/poseidon/api/dto"
|
"gitlab.hpi.de/codeocean/codemoon/poseidon/api/dto"
|
||||||
|
"gitlab.hpi.de/codeocean/codemoon/poseidon/logging"
|
||||||
"gitlab.hpi.de/codeocean/codemoon/poseidon/store"
|
"gitlab.hpi.de/codeocean/codemoon/poseidon/store"
|
||||||
"sync"
|
"sync"
|
||||||
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var log = logging.GetLogger("runner")
|
||||||
|
|
||||||
// Status is the type for the status of a Runner.
|
// Status is the type for the status of a Runner.
|
||||||
type Status string
|
type Status string
|
||||||
|
|
||||||
@ -46,6 +50,10 @@ type Runner interface {
|
|||||||
|
|
||||||
// DeleteExecution deletes the execution of the runner with the specified id.
|
// DeleteExecution deletes the execution of the runner with the specified id.
|
||||||
DeleteExecution(ExecutionId)
|
DeleteExecution(ExecutionId)
|
||||||
|
|
||||||
|
// Execute runs one of the runners Executions by it's id
|
||||||
|
// ok will be false if the runner has no execution with the provided id
|
||||||
|
Execute(ExecutionId) (err error, ok bool)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ExerciseRunner is an abstraction to communicate with Nomad allocations.
|
// ExerciseRunner is an abstraction to communicate with Nomad allocations.
|
||||||
@ -120,6 +128,19 @@ func (r *ExerciseRunner) DeleteExecution(id ExecutionId) {
|
|||||||
delete(r.executions, id)
|
delete(r.executions, id)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *ExerciseRunner) Execute(id ExecutionId) (err error, ok bool) {
|
||||||
|
execution, ok := r.Execution(id)
|
||||||
|
if !ok {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), time.Duration(execution.TimeLimit)*time.Second)
|
||||||
|
defer cancel()
|
||||||
|
log.WithField("Context", ctx).Printf("ToDo: Running execution")
|
||||||
|
// ToDo: Implement command execution
|
||||||
|
// r.nomadApiClient.ExecuteCommand(r.id, ctx, )
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// NewContext creates a context containing a runner.
|
// NewContext creates a context containing a runner.
|
||||||
func NewContext(ctx context.Context, runner Runner) context.Context {
|
func NewContext(ctx context.Context, runner Runner) context.Context {
|
||||||
return context.WithValue(ctx, runnerContextKey, runner)
|
return context.WithValue(ctx, runnerContextKey, runner)
|
||||||
|
Reference in New Issue
Block a user