179 lines
4.7 KiB
Go
179 lines
4.7 KiB
Go
// Copyright 2014 Google Inc. All Rights Reserved.
|
|
//
|
|
// 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 api provides a handler for /api/
|
|
package api
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"path"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/golang/glog"
|
|
"github.com/google/cadvisor/info"
|
|
"github.com/google/cadvisor/manager"
|
|
)
|
|
|
|
const (
|
|
apiResource = "/api/"
|
|
containersApi = "containers"
|
|
subcontainersApi = "subcontainers"
|
|
machineApi = "machine"
|
|
|
|
version1_0 = "v1.0"
|
|
version1_1 = "v1.1"
|
|
)
|
|
|
|
var supportedApiVersions map[string]struct{} = map[string]struct{}{
|
|
version1_0: {},
|
|
version1_1: {},
|
|
}
|
|
|
|
func RegisterHandlers(m manager.Manager) error {
|
|
http.HandleFunc(apiResource, func(w http.ResponseWriter, r *http.Request) {
|
|
if err := handleRequest(m, w, r); err != nil {
|
|
fmt.Fprintf(w, "%s", err)
|
|
}
|
|
})
|
|
|
|
return nil
|
|
}
|
|
|
|
func handleRequest(m manager.Manager, w http.ResponseWriter, r *http.Request) error {
|
|
start := time.Now()
|
|
|
|
request := r.URL.Path
|
|
requestElements := strings.Split(r.URL.Path, "/")
|
|
|
|
// Verify that we have all the elements we expect:
|
|
// <empty>/api/<version>/<request type>[/<args...>]
|
|
// [0] [1] [2] [3] [4...]
|
|
if len(requestElements) < 4 {
|
|
return fmt.Errorf("incomplete API request %q", request)
|
|
}
|
|
|
|
// Get all the element parts.
|
|
emptyElement := requestElements[0]
|
|
apiElement := requestElements[1]
|
|
version := requestElements[2]
|
|
requestType := requestElements[3]
|
|
requestArgs := []string{}
|
|
if len(requestElements) > 4 {
|
|
requestArgs = requestElements[4:]
|
|
}
|
|
|
|
// Check elements.
|
|
if len(emptyElement) != 0 {
|
|
return fmt.Errorf("unexpected API request format %q", request)
|
|
}
|
|
if apiElement != "api" {
|
|
return fmt.Errorf("invalid API request format %q", request)
|
|
}
|
|
if _, ok := supportedApiVersions[version]; !ok {
|
|
return fmt.Errorf("unsupported API version %q", version)
|
|
}
|
|
|
|
switch {
|
|
case requestType == machineApi:
|
|
glog.V(2).Infof("Api - Machine")
|
|
|
|
// Get the MachineInfo
|
|
machineInfo, err := m.GetMachineInfo()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
out, err := json.Marshal(machineInfo)
|
|
if err != nil {
|
|
fmt.Fprintf(w, "Failed to marshall MachineInfo with error: %s", err)
|
|
}
|
|
w.Write(out)
|
|
case requestType == containersApi:
|
|
// The container name is the path after the requestType.
|
|
containerName := path.Join("/", strings.Join(requestArgs, "/"))
|
|
|
|
glog.V(2).Infof("Api - Container(%s)", containerName)
|
|
|
|
// Get the query request.
|
|
query, err := getContainerInfoRequest(r.Body)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Get the container.
|
|
cont, err := m.GetContainerInfo(containerName, query)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to get container %q with error: %s", containerName, err)
|
|
}
|
|
|
|
// Only output the container as JSON.
|
|
out, err := json.Marshal(cont)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to marshall container %q with error: %s", containerName, err)
|
|
}
|
|
w.Write(out)
|
|
case requestType == subcontainersApi:
|
|
if version == version1_0 {
|
|
return fmt.Errorf("request type of %q not supported in API version %q", requestType, version)
|
|
}
|
|
|
|
// The container name is the path after the requestType.
|
|
containerName := path.Join("/", strings.Join(requestArgs, "/"))
|
|
|
|
glog.V(2).Infof("Api - Subcontainers(%s)", containerName)
|
|
|
|
// Get the query request.
|
|
query, err := getContainerInfoRequest(r.Body)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Get the subcontainers.
|
|
containers, err := m.SubcontainersInfo(containerName, query)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to get subcontainers for container %q with error: %s", containerName, err)
|
|
}
|
|
|
|
// Only output the containers as JSON.
|
|
out, err := json.Marshal(containers)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to marshall container %q with error: %s", containerName, err)
|
|
}
|
|
w.Write(out)
|
|
default:
|
|
return fmt.Errorf("unknown API request type %q", requestType)
|
|
}
|
|
|
|
glog.V(2).Infof("Request took %s", time.Since(start))
|
|
return nil
|
|
}
|
|
|
|
func getContainerInfoRequest(body io.ReadCloser) (*info.ContainerInfoRequest, error) {
|
|
var query info.ContainerInfoRequest
|
|
|
|
// Default stats and samples is 64.
|
|
query.NumStats = 64
|
|
|
|
decoder := json.NewDecoder(body)
|
|
if err := decoder.Decode(&query); err != nil && err != io.EOF {
|
|
return nil, fmt.Errorf("unable to decode the json value: %s", err)
|
|
}
|
|
|
|
return &query, nil
|
|
}
|