notebooks/workspaces/backend/internal/server/server.go

104 lines
2.6 KiB
Go

/*
Copyright 2024.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package server
import (
"context"
"errors"
"fmt"
"log/slog"
"net"
"net/http"
"time"
ctrl "sigs.k8s.io/controller-runtime"
"github.com/kubeflow/notebooks/workspaces/backend/api"
)
const (
maxHeaderBytes = 1 << 17 // 128 KiB - default is 1MiB
maxBodyBytes = 1 << 22 // 4 MiB - default is unlimited
)
type Server struct {
logger *slog.Logger
listener net.Listener
server *http.Server
}
func NewServer(app *api.App, logger *slog.Logger) (*Server, error) {
listener, err := net.Listen("tcp", fmt.Sprintf(":%d", app.Config.Port))
if err != nil {
return nil, fmt.Errorf("failed to create listener: %w", err)
}
svc := &http.Server{
Addr: fmt.Sprintf(":%d", app.Config.Port),
Handler: http.MaxBytesHandler(app.Routes(), maxBodyBytes),
IdleTimeout: 90 * time.Second, // matches http.DefaultTransport keep-alive timeout
ReadTimeout: 32 * time.Second,
ReadHeaderTimeout: 32 * time.Second,
WriteTimeout: 32 * time.Second,
MaxHeaderBytes: maxHeaderBytes,
ErrorLog: slog.NewLogLogger(logger.Handler(), slog.LevelError),
}
svr := &Server{
logger: logger,
listener: listener,
server: svc,
}
return svr, nil
}
// SetupWithManager sets up the app with a controller-runtime manager
func (s *Server) SetupWithManager(mgr ctrl.Manager) error {
err := mgr.Add(s)
if err != nil {
return err
}
return nil
}
// Start starts the server
// Blocks until the context is canceled
func (s *Server) Start(ctx context.Context) error {
serverShutdown := make(chan struct{})
go func() {
<-ctx.Done()
s.logger.Info("shutting down server")
if err := s.server.Shutdown(context.Background()); err != nil {
s.logger.Error("error shutting down server", "error", err)
}
close(serverShutdown)
}()
s.logger.Info("starting server", "addr", s.server.Addr)
if err := s.server.Serve(s.listener); err != nil && !errors.Is(err, http.ErrServerClosed) {
return err
}
<-serverShutdown
return nil
}
// NeedLeaderElection returns false because this app does not need leader election from the manager
func (s *Server) NeedLeaderElection() bool {
return false
}