moby--moby/api.go

372 lines
10 KiB
Go
Raw Normal View History

2013-04-11 02:48:21 +00:00
package docker
import (
2013-04-22 21:37:22 +00:00
_"bytes"
"encoding/json"
"fmt"
2013-04-11 02:48:21 +00:00
"github.com/gorilla/mux"
"io/ioutil"
2013-04-18 16:56:22 +00:00
"log"
2013-04-11 02:48:21 +00:00
"net/http"
"os"
"runtime"
2013-04-22 16:17:47 +00:00
"strconv"
"strings"
2013-04-18 16:56:22 +00:00
"time"
2013-04-11 02:48:21 +00:00
)
func ListenAndServe(addr string, rtime *Runtime) error {
r := mux.NewRouter()
log.Printf("Listening for HTTP on %s\n", addr)
2013-04-11 02:48:21 +00:00
2013-04-22 16:17:47 +00:00
r.Path("/version").Methods("GET").HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
log.Println(r.RequestURI)
m := VersionOut{VERSION, GIT_COMMIT, NO_MEMORY_LIMIT}
b, err := json.Marshal(m)
if err != nil {
2013-04-22 16:17:47 +00:00
http.Error(w, err.Error(), http.StatusInternalServerError)
} else {
w.Write(b)
}
})
2013-04-11 02:48:21 +00:00
2013-04-22 16:17:47 +00:00
r.Path("/containers/{name:.*}/kill").Methods("POST").HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
log.Println(r.RequestURI)
vars := mux.Vars(r)
name := vars["name"]
if container := rtime.Get(name); container != nil {
if err := container.Kill(); err != nil {
http.Error(w, "Error restarting container "+name+": "+err.Error(), http.StatusInternalServerError)
return
}
} else {
http.Error(w, "No such container: "+name, http.StatusInternalServerError)
return
}
w.WriteHeader(200)
2013-04-18 16:56:22 +00:00
})
2013-04-22 16:17:47 +00:00
r.Path("/images").Methods("GET").HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
log.Println(r.RequestURI)
All := r.Form.Get("all")
NameFilter := r.Form.Get("filter")
Quiet := r.Form.Get("quiet")
2013-04-18 16:56:22 +00:00
var allImages map[string]*Image
var err error
2013-04-22 16:17:47 +00:00
if All == "true" {
allImages, err = rtime.graph.Map()
2013-04-18 16:56:22 +00:00
} else {
allImages, err = rtime.graph.Heads()
2013-04-18 16:56:22 +00:00
}
if err != nil {
w.WriteHeader(500)
return
}
var outs []ImagesOut
for name, repository := range rtime.repositories.Repositories {
2013-04-22 16:17:47 +00:00
if NameFilter != "" && name != NameFilter {
2013-04-18 16:56:22 +00:00
continue
}
for tag, id := range repository {
var out ImagesOut
image, err := rtime.graph.Get(id)
2013-04-18 16:56:22 +00:00
if err != nil {
log.Printf("Warning: couldn't load %s from %s/%s: %s", id, name, tag, err)
continue
}
delete(allImages, id)
2013-04-22 16:17:47 +00:00
if Quiet != "true" {
2013-04-18 16:56:22 +00:00
out.Repository = name
out.Tag = tag
out.Id = TruncateId(id)
out.Created = HumanDuration(time.Now().Sub(image.Created)) + " ago"
} else {
out.Id = image.ShortId()
}
outs = append(outs, out)
}
}
// Display images which aren't part of a
2013-04-22 16:17:47 +00:00
if NameFilter == "" {
2013-04-18 16:56:22 +00:00
for id, image := range allImages {
var out ImagesOut
2013-04-22 16:17:47 +00:00
if Quiet != "true" {
2013-04-18 16:56:22 +00:00
out.Repository = "<none>"
out.Tag = "<none>"
out.Id = TruncateId(id)
out.Created = HumanDuration(time.Now().Sub(image.Created)) + " ago"
} else {
out.Id = image.ShortId()
}
outs = append(outs, out)
}
}
b, err := json.Marshal(outs)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
} else {
w.Write(b)
}
})
2013-04-22 16:17:47 +00:00
r.Path("/info").Methods("GET").HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
log.Println(r.RequestURI)
images, _ := rtime.graph.All()
var imgcount int
if images == nil {
imgcount = 0
} else {
imgcount = len(images)
}
var out InfoOut
out.Containers = len(rtime.List())
out.Version = VERSION
out.Images = imgcount
if os.Getenv("DEBUG") == "1" {
out.Debug = true
out.NFd = getTotalUsedFds()
out.NGoroutines = runtime.NumGoroutine()
}
b, err := json.Marshal(out)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
} else {
w.Write(b)
}
})
2013-04-22 16:17:47 +00:00
r.Path("/images/{name:.*}/history").Methods("GET").HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
log.Println(r.RequestURI)
2013-04-22 16:17:47 +00:00
vars := mux.Vars(r)
name := vars["name"]
2013-04-22 16:17:47 +00:00
image, err := rtime.repositories.LookupImage(name)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
var outs []HistoryOut
err = image.WalkHistory(func(img *Image) error {
var out HistoryOut
out.Id = rtime.repositories.ImageName(img.ShortId())
out.Created = HumanDuration(time.Now().Sub(img.Created)) + " ago"
out.CreatedBy = strings.Join(img.ContainerConfig.Cmd, " ")
return nil
})
b, err := json.Marshal(outs)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
} else {
w.Write(b)
}
})
2013-04-22 16:17:47 +00:00
r.Path("/containers/{name:.*}/logs").Methods("GET").HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
log.Println(r.RequestURI)
vars := mux.Vars(r)
name := vars["name"]
2013-04-22 16:17:47 +00:00
if container := rtime.Get(name); container != nil {
var out LogsOut
logStdout, err := container.ReadLog("stdout")
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
logStderr, err := container.ReadLog("stderr")
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
stdout, errStdout := ioutil.ReadAll(logStdout)
if errStdout != nil {
http.Error(w, errStdout.Error(), http.StatusInternalServerError)
return
} else {
out.Stdout = fmt.Sprintf("%s", stdout)
}
stderr, errStderr := ioutil.ReadAll(logStderr)
if errStderr != nil {
http.Error(w, errStderr.Error(), http.StatusInternalServerError)
return
} else {
out.Stderr = fmt.Sprintf("%s", stderr)
}
b, err := json.Marshal(out)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
} else {
w.Write(b)
}
} else {
2013-04-22 16:17:47 +00:00
http.Error(w, "No such container: "+name, http.StatusInternalServerError)
}
})
2013-04-22 16:17:47 +00:00
r.Path("/containers").Methods("GET").HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
log.Println(r.RequestURI)
All := r.Form.Get("all")
NoTrunc := r.Form.Get("notrunc")
Quiet := r.Form.Get("quiet")
Last := r.Form.Get("n")
n, err := strconv.Atoi(Last)
if err != nil {
n = -1
}
var outs []PsOut
for i, container := range rtime.List() {
2013-04-22 16:17:47 +00:00
if !container.State.Running && All != "true" && n == -1 {
continue
}
2013-04-22 16:17:47 +00:00
if i == n {
break
}
var out PsOut
out.Id = container.ShortId()
2013-04-22 16:17:47 +00:00
if Quiet != "true" {
command := fmt.Sprintf("%s %s", container.Path, strings.Join(container.Args, " "))
2013-04-22 16:17:47 +00:00
if NoTrunc != "true" {
command = Trunc(command, 20)
}
out.Image = rtime.repositories.ImageName(container.Image)
out.Command = command
out.Created = HumanDuration(time.Now().Sub(container.Created)) + " ago"
out.Status = container.State.String()
}
outs = append(outs, out)
}
b, err := json.Marshal(outs)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
} else {
w.Write(b)
}
})
2013-04-22 21:37:22 +00:00
r.Path("/pull").Methods("GET", "POST").HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
var in PullIn
//json.NewDecoder(r.Body).Decode(&in)
in.Name = "base"
hj, ok := w.(http.Hijacker)
if !ok {
http.Error(w, "webserver doesn't support hijacking", http.StatusInternalServerError)
return
}
conn, bufrw, err := hj.Hijack()
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
// Don't forget to close the connection:
defer conn.Close()
if rtime.graph.LookupRemoteImage(in.Name, rtime.authConfig) {
if err := rtime.graph.PullImage(bufrw, in.Name, rtime.authConfig); err != nil {
//http.Error(w, err.Error(), http.StatusInternalServerError)
}
return
}
if err := rtime.graph.PullRepository(bufrw, in.Name, "", rtime.repositories, rtime.authConfig); err != nil {
//http.Error(w, err.Error(), http.StatusInternalServerError)
}
return
})
2013-04-22 16:17:47 +00:00
r.Path("/containers/{name:.*}/restart").Methods("POST").HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
log.Println(r.RequestURI)
vars := mux.Vars(r)
name := vars["name"]
if container := rtime.Get(name); container != nil {
if err := container.Restart(); err != nil {
http.Error(w, "Error restarting container "+name+": "+err.Error(), http.StatusInternalServerError)
return
}
} else {
2013-04-22 16:17:47 +00:00
http.Error(w, "No such container: "+name, http.StatusInternalServerError)
return
}
2013-04-22 16:17:47 +00:00
w.WriteHeader(200)
})
2013-04-22 16:17:47 +00:00
r.Path("/containers/{name:.*}").Methods("DELETE").HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
log.Println(r.RequestURI)
vars := mux.Vars(r)
name := vars["name"]
if container := rtime.Get(name); container != nil {
if err := rtime.Destroy(container); err != nil {
http.Error(w, "Error destroying container "+name+": "+err.Error(), http.StatusInternalServerError)
return
}
} else {
2013-04-22 16:17:47 +00:00
http.Error(w, "No such container: "+name, http.StatusInternalServerError)
return
}
2013-04-22 16:17:47 +00:00
w.WriteHeader(200)
})
2013-04-22 16:17:47 +00:00
r.Path("/images/{name:.*}").Methods("DELETE").HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
log.Println(r.RequestURI)
vars := mux.Vars(r)
name := vars["name"]
img, err := rtime.repositories.LookupImage(name)
if err != nil {
2013-04-22 16:17:47 +00:00
http.Error(w, "No such image: "+name, http.StatusInternalServerError)
return
2013-04-22 16:17:47 +00:00
} else {
if err := rtime.graph.Delete(img.Id); err != nil {
http.Error(w, "Error deleting image "+name+": "+err.Error(), http.StatusInternalServerError)
return
}
}
2013-04-22 16:17:47 +00:00
w.WriteHeader(200)
})
2013-04-22 16:17:47 +00:00
r.Path("/containers/{name:.*}/start").Methods("POST").HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
log.Println(r.RequestURI)
vars := mux.Vars(r)
name := vars["name"]
if container := rtime.Get(name); container != nil {
if err := container.Start(); err != nil {
http.Error(w, "Error starting container "+name+": "+err.Error(), http.StatusInternalServerError)
return
}
} else {
2013-04-22 16:17:47 +00:00
http.Error(w, "No such container: "+name, http.StatusInternalServerError)
return
}
2013-04-22 16:17:47 +00:00
w.WriteHeader(200)
})
2013-04-22 16:17:47 +00:00
r.Path("/containers/{name:.*}/stop").Methods("POST").HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
log.Println(r.RequestURI)
vars := mux.Vars(r)
name := vars["name"]
if container := rtime.Get(name); container != nil {
if err := container.Stop(); err != nil {
http.Error(w, "Error stopping container "+name+": "+err.Error(), http.StatusInternalServerError)
return
}
2013-04-18 16:56:22 +00:00
} else {
2013-04-22 16:17:47 +00:00
http.Error(w, "No such container: "+name, http.StatusInternalServerError)
return
2013-04-18 16:56:22 +00:00
}
2013-04-22 16:17:47 +00:00
w.WriteHeader(200)
})
2013-04-11 02:48:21 +00:00
return http.ListenAndServe(addr, r)
2013-04-11 02:48:21 +00:00
}