177 lines
5.0 KiB
Go
177 lines
5.0 KiB
Go
package vkubelet
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"os"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/Sirupsen/logrus"
|
|
"github.com/gorilla/mux"
|
|
"github.com/pkg/errors"
|
|
"github.com/virtual-kubelet/virtual-kubelet/log"
|
|
"k8s.io/kubernetes/pkg/kubelet/server/remotecommand"
|
|
)
|
|
|
|
var p Provider
|
|
var r mux.Router
|
|
|
|
func loggingContext(r *http.Request) context.Context {
|
|
ctx := r.Context()
|
|
logger := log.G(ctx).WithFields(logrus.Fields{
|
|
"uri": r.RequestURI,
|
|
"vars": mux.Vars(r),
|
|
})
|
|
return log.WithLogger(ctx, logger)
|
|
}
|
|
|
|
func NotFound(w http.ResponseWriter, r *http.Request) {
|
|
logger := log.G(loggingContext(r))
|
|
log.Trace(logger, "404 request not found")
|
|
http.Error(w, "404 request not found", http.StatusNotFound)
|
|
}
|
|
|
|
func ApiserverStart(provider Provider, metricsAddr string) {
|
|
p = provider
|
|
certFilePath := os.Getenv("APISERVER_CERT_LOCATION")
|
|
keyFilePath := os.Getenv("APISERVER_KEY_LOCATION")
|
|
port := os.Getenv("KUBELET_PORT")
|
|
addr := fmt.Sprintf(":%s", port)
|
|
|
|
r := mux.NewRouter()
|
|
r.HandleFunc("/containerLogs/{namespace}/{pod}/{container}", ApiServerHandler).Methods("GET")
|
|
r.HandleFunc("/exec/{namespace}/{pod}/{container}", ApiServerHandlerExec).Methods("POST")
|
|
r.NotFoundHandler = http.HandlerFunc(NotFound)
|
|
|
|
if metricsAddr != "" {
|
|
go MetricsServerStart(metricsAddr)
|
|
} else {
|
|
log.G(context.TODO()).Info("Skipping metrics server startup since no address was provided")
|
|
}
|
|
|
|
if err := http.ListenAndServeTLS(addr, certFilePath, keyFilePath, r); err != nil {
|
|
log.G(context.TODO()).WithError(err).Error("error setting up http server")
|
|
}
|
|
}
|
|
|
|
// MetricsServerStart starts an HTTP server on the provided addr for serving the kubelset summary stats API.
|
|
// TLS is never enabled on this endpoint.
|
|
func MetricsServerStart(addr string) {
|
|
r := mux.NewRouter()
|
|
r.HandleFunc("/stats/summary", MetricsSummaryHandler).Methods("GET")
|
|
r.HandleFunc("/stats/summary/", MetricsSummaryHandler).Methods("GET")
|
|
r.NotFoundHandler = http.HandlerFunc(NotFound)
|
|
if err := http.ListenAndServe(addr, r); err != nil {
|
|
log.G(context.TODO()).WithError(err).Error("Error starting http server")
|
|
}
|
|
}
|
|
|
|
// MetricsSummaryHandler is an HTTP handler for implementing the kubelet summary stats endpoint
|
|
func MetricsSummaryHandler(w http.ResponseWriter, req *http.Request) {
|
|
ctx := loggingContext(req)
|
|
|
|
mp, ok := p.(MetricsProvider)
|
|
if !ok {
|
|
log.G(ctx).Debug("stats not implemented for provider")
|
|
http.Error(w, "not implememnted", http.StatusNotImplemented)
|
|
return
|
|
}
|
|
|
|
stats, err := mp.GetStatsSummary(req.Context())
|
|
if err != nil {
|
|
if errors.Cause(err) == context.Canceled {
|
|
return
|
|
}
|
|
log.G(ctx).Error("Error getting stats from provider:", err)
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
b, err := json.Marshal(stats)
|
|
if err != nil {
|
|
log.G(ctx).WithError(err).Error("Could not marshal stats")
|
|
http.Error(w, "could not marshal stats: "+err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
if _, err := w.Write(b); err != nil {
|
|
log.G(ctx).WithError(err).Debug("Could not write to client")
|
|
}
|
|
}
|
|
|
|
func ApiServerHandler(w http.ResponseWriter, req *http.Request) {
|
|
vars := mux.Vars(req)
|
|
if len(vars) != 3 {
|
|
NotFound(w, req)
|
|
return
|
|
}
|
|
|
|
ctx := loggingContext(req)
|
|
|
|
namespace := vars["namespace"]
|
|
pod := vars["pod"]
|
|
container := vars["container"]
|
|
tail := 10
|
|
q := req.URL.Query()
|
|
|
|
if queryTail := q.Get("tailLines"); queryTail != "" {
|
|
t, err := strconv.Atoi(queryTail)
|
|
if err != nil {
|
|
logger := log.G(context.TODO()).WithError(err)
|
|
log.Trace(logger, "could not parse tailLines")
|
|
http.Error(w, fmt.Sprintf("could not parse \"tailLines\": %v", err), http.StatusBadRequest)
|
|
return
|
|
}
|
|
tail = t
|
|
}
|
|
|
|
podsLogs, err := p.GetContainerLogs(namespace, pod, container, tail)
|
|
if err != nil {
|
|
log.G(ctx).WithError(err).Error("error getting container logs")
|
|
http.Error(w, fmt.Sprintf("error while getting container logs: %v", err), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
if _, err := io.WriteString(w, podsLogs); err != nil {
|
|
log.G(ctx).WithError(err).Warn("error writing response to client")
|
|
}
|
|
}
|
|
|
|
func ApiServerHandlerExec(w http.ResponseWriter, req *http.Request) {
|
|
vars := mux.Vars(req)
|
|
|
|
namespace := vars["namespace"]
|
|
pod := vars["pod"]
|
|
container := vars["container"]
|
|
|
|
supportedStreamProtocols := strings.Split(req.Header.Get("X-Stream-Protocol-Version"), ",")
|
|
|
|
q := req.URL.Query()
|
|
command := q["command"]
|
|
|
|
// streamOpts := &remotecommand.Options{
|
|
// Stdin: (q.Get("input") == "1"),
|
|
// Stdout: (q.Get("output") == "1"),
|
|
// Stderr: (q.Get("error") == "1"),
|
|
// TTY: (q.Get("tty") == "1"),
|
|
// }
|
|
|
|
// TODO: tty flag causes remotecommand.createStreams to wait for the wrong number of streams
|
|
streamOpts := &remotecommand.Options{
|
|
Stdin: true,
|
|
Stdout: true,
|
|
Stderr: true,
|
|
TTY: false,
|
|
}
|
|
|
|
idleTimeout := time.Second * 30
|
|
streamCreationTimeout := time.Second * 30
|
|
|
|
remotecommand.ServeExec(w, req, p, fmt.Sprintf("%s-%s", namespace, pod), "", container, command, streamOpts, idleTimeout, streamCreationTimeout, supportedStreamProtocols)
|
|
}
|