b78e4216 |
// +build !windows
package main
import (
"encoding/json"
"fmt"
"io"
"io/ioutil"
"net/http"
"net/http/httptest"
"os"
"strings"
"github.com/docker/docker/daemon/graphdriver"
"github.com/docker/docker/daemon/graphdriver/vfs" |
48de91a3 |
"github.com/docker/docker/integration-cli/daemon" |
b78e4216 |
"github.com/docker/docker/pkg/archive" |
de806a67 |
"github.com/docker/docker/pkg/plugins" |
b78e4216 |
"github.com/go-check/check"
)
func init() {
check.Suite(&DockerExternalGraphdriverSuite{
ds: &DockerSuite{},
})
}
type DockerExternalGraphdriverSuite struct { |
de806a67 |
server *httptest.Server
jserver *httptest.Server
ds *DockerSuite |
48de91a3 |
d *daemon.Daemon |
de806a67 |
ec map[string]*graphEventsCounter |
b78e4216 |
}
type graphEventsCounter struct {
activations int
creations int
removals int
gets int
puts int
stats int
cleanups int
exists int
init int
metadata int
diff int
applydiff int
changes int
diffsize int
}
func (s *DockerExternalGraphdriverSuite) SetUpTest(c *check.C) { |
48de91a3 |
s.d = daemon.New(c, dockerBinary, dockerdBinary, daemon.Config{ |
c8016e66 |
Experimental: testEnv.ExperimentalDaemon(), |
48de91a3 |
}) |
b78e4216 |
}
|
2938dce7 |
func (s *DockerExternalGraphdriverSuite) OnTimeout(c *check.C) {
s.d.DumpStackAndQuit()
}
|
b78e4216 |
func (s *DockerExternalGraphdriverSuite) TearDownTest(c *check.C) { |
5890091c |
if s.d != nil { |
c502fb49 |
s.d.Stop(c) |
5890091c |
s.ds.TearDownTest(c)
} |
b78e4216 |
}
func (s *DockerExternalGraphdriverSuite) SetUpSuite(c *check.C) { |
de806a67 |
s.ec = make(map[string]*graphEventsCounter)
s.setUpPluginViaSpecFile(c)
s.setUpPluginViaJSONFile(c)
}
func (s *DockerExternalGraphdriverSuite) setUpPluginViaSpecFile(c *check.C) { |
b78e4216 |
mux := http.NewServeMux()
s.server = httptest.NewServer(mux)
|
de806a67 |
s.setUpPlugin(c, "test-external-graph-driver", "spec", mux, []byte(s.server.URL))
}
func (s *DockerExternalGraphdriverSuite) setUpPluginViaJSONFile(c *check.C) {
mux := http.NewServeMux()
s.jserver = httptest.NewServer(mux)
|
f3711704 |
p := plugins.NewLocalPlugin("json-external-graph-driver", s.jserver.URL) |
de806a67 |
b, err := json.Marshal(p)
c.Assert(err, check.IsNil)
s.setUpPlugin(c, "json-external-graph-driver", "json", mux, b)
}
func (s *DockerExternalGraphdriverSuite) setUpPlugin(c *check.C, name string, ext string, mux *http.ServeMux, b []byte) { |
b78e4216 |
type graphDriverRequest struct {
ID string `json:",omitempty"`
Parent string `json:",omitempty"`
MountLabel string `json:",omitempty"` |
ef5bfad3 |
ReadOnly bool `json:",omitempty"` |
b78e4216 |
}
type graphDriverResponse struct {
Err error `json:",omitempty"`
Dir string `json:",omitempty"`
Exists bool `json:",omitempty"`
Status [][2]string `json:",omitempty"`
Metadata map[string]string `json:",omitempty"`
Changes []archive.Change `json:",omitempty"`
Size int64 `json:",omitempty"`
}
respond := func(w http.ResponseWriter, data interface{}) { |
39bcaee4 |
w.Header().Set("Content-Type", "application/vnd.docker.plugins.v1+json") |
b78e4216 |
switch t := data.(type) {
case error: |
442b4562 |
fmt.Fprintln(w, fmt.Sprintf(`{"Err": %q}`, t.Error())) |
b78e4216 |
case string:
fmt.Fprintln(w, t)
default:
json.NewEncoder(w).Encode(&data)
}
}
|
442b4562 |
decReq := func(b io.ReadCloser, out interface{}, w http.ResponseWriter) error {
defer b.Close()
if err := json.NewDecoder(b).Decode(&out); err != nil {
http.Error(w, fmt.Sprintf("error decoding json: %s", err.Error()), 500)
}
return nil
}
|
de806a67 |
base, err := ioutil.TempDir("", name) |
b78e4216 |
c.Assert(err, check.IsNil) |
442b4562 |
vfsProto, err := vfs.Init(base, []string{}, nil, nil) |
7058ec27 |
c.Assert(err, check.IsNil, check.Commentf("error initializing graph driver")) |
442b4562 |
driver := graphdriver.NewNaiveDiffDriver(vfsProto, nil, nil) |
b78e4216 |
|
de806a67 |
s.ec[ext] = &graphEventsCounter{} |
b78e4216 |
mux.HandleFunc("/Plugin.Activate", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].activations++ |
b78e4216 |
respond(w, `{"Implements": ["GraphDriver"]}`)
})
mux.HandleFunc("/GraphDriver.Init", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].init++ |
b78e4216 |
respond(w, "{}")
})
|
ef5bfad3 |
mux.HandleFunc("/GraphDriver.CreateReadWrite", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].creations++ |
ef5bfad3 |
var req graphDriverRequest
if err := decReq(r.Body, &req, w); err != nil {
return
} |
b937aa8e |
if err := driver.CreateReadWrite(req.ID, req.Parent, nil); err != nil { |
ef5bfad3 |
respond(w, err)
return
}
respond(w, "{}")
})
|
b78e4216 |
mux.HandleFunc("/GraphDriver.Create", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].creations++ |
b78e4216 |
var req graphDriverRequest |
442b4562 |
if err := decReq(r.Body, &req, w); err != nil { |
b78e4216 |
return
} |
b937aa8e |
if err := driver.Create(req.ID, req.Parent, nil); err != nil { |
b78e4216 |
respond(w, err)
return
}
respond(w, "{}")
})
mux.HandleFunc("/GraphDriver.Remove", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].removals++ |
b78e4216 |
var req graphDriverRequest |
442b4562 |
if err := decReq(r.Body, &req, w); err != nil { |
b78e4216 |
return
}
if err := driver.Remove(req.ID); err != nil {
respond(w, err)
return
}
respond(w, "{}")
})
mux.HandleFunc("/GraphDriver.Get", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].gets++ |
b78e4216 |
var req graphDriverRequest |
442b4562 |
if err := decReq(r.Body, &req, w); err != nil {
return |
b78e4216 |
}
dir, err := driver.Get(req.ID, req.MountLabel)
if err != nil {
respond(w, err)
return
}
respond(w, &graphDriverResponse{Dir: dir})
})
mux.HandleFunc("/GraphDriver.Put", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].puts++ |
b78e4216 |
var req graphDriverRequest |
442b4562 |
if err := decReq(r.Body, &req, w); err != nil { |
b78e4216 |
return
}
if err := driver.Put(req.ID); err != nil {
respond(w, err)
return
}
respond(w, "{}")
})
mux.HandleFunc("/GraphDriver.Exists", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].exists++ |
b78e4216 |
var req graphDriverRequest |
442b4562 |
if err := decReq(r.Body, &req, w); err != nil { |
b78e4216 |
return
}
respond(w, &graphDriverResponse{Exists: driver.Exists(req.ID)})
})
mux.HandleFunc("/GraphDriver.Status", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].stats++ |
442b4562 |
respond(w, &graphDriverResponse{Status: driver.Status()}) |
b78e4216 |
})
mux.HandleFunc("/GraphDriver.Cleanup", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].cleanups++ |
b78e4216 |
err := driver.Cleanup()
if err != nil {
respond(w, err)
return
}
respond(w, `{}`)
})
mux.HandleFunc("/GraphDriver.GetMetadata", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].metadata++ |
b78e4216 |
var req graphDriverRequest |
442b4562 |
if err := decReq(r.Body, &req, w); err != nil { |
b78e4216 |
return
}
data, err := driver.GetMetadata(req.ID)
if err != nil {
respond(w, err)
return
}
respond(w, &graphDriverResponse{Metadata: data})
})
mux.HandleFunc("/GraphDriver.Diff", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].diff++ |
b78e4216 |
var req graphDriverRequest |
442b4562 |
if err := decReq(r.Body, &req, w); err != nil { |
b78e4216 |
return
}
diff, err := driver.Diff(req.ID, req.Parent)
if err != nil {
respond(w, err)
return
}
io.Copy(w, diff)
})
mux.HandleFunc("/GraphDriver.Changes", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].changes++ |
b78e4216 |
var req graphDriverRequest |
442b4562 |
if err := decReq(r.Body, &req, w); err != nil { |
b78e4216 |
return
}
changes, err := driver.Changes(req.ID, req.Parent)
if err != nil {
respond(w, err)
return
}
respond(w, &graphDriverResponse{Changes: changes})
})
mux.HandleFunc("/GraphDriver.ApplyDiff", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].applydiff++ |
7781a1bf |
diff := r.Body |
442b4562 |
defer r.Body.Close()
|
b78e4216 |
id := r.URL.Query().Get("id")
parent := r.URL.Query().Get("parent")
|
442b4562 |
if id == "" {
http.Error(w, fmt.Sprintf("missing id"), 409)
}
size, err := driver.ApplyDiff(id, parent, diff) |
b78e4216 |
if err != nil {
respond(w, err)
return
}
respond(w, &graphDriverResponse{Size: size})
})
mux.HandleFunc("/GraphDriver.DiffSize", func(w http.ResponseWriter, r *http.Request) { |
de806a67 |
s.ec[ext].diffsize++ |
b78e4216 |
var req graphDriverRequest |
442b4562 |
if err := decReq(r.Body, &req, w); err != nil { |
b78e4216 |
return
}
size, err := driver.DiffSize(req.ID, req.Parent)
if err != nil {
respond(w, err)
return
}
respond(w, &graphDriverResponse{Size: size})
})
|
7058ec27 |
err = os.MkdirAll("/etc/docker/plugins", 0755)
c.Assert(err, check.IsNil, check.Commentf("error creating /etc/docker/plugins")) |
b78e4216 |
|
de806a67 |
specFile := "/etc/docker/plugins/" + name + "." + ext
err = ioutil.WriteFile(specFile, b, 0644)
c.Assert(err, check.IsNil, check.Commentf("error writing to %s", specFile)) |
b78e4216 |
}
func (s *DockerExternalGraphdriverSuite) TearDownSuite(c *check.C) {
s.server.Close() |
de806a67 |
s.jserver.Close() |
b78e4216 |
|
7058ec27 |
err := os.RemoveAll("/etc/docker/plugins")
c.Assert(err, check.IsNil, check.Commentf("error removing /etc/docker/plugins")) |
b78e4216 |
}
func (s *DockerExternalGraphdriverSuite) TestExternalGraphDriver(c *check.C) { |
7781a1bf |
testRequires(c, ExperimentalDaemon)
|
de806a67 |
s.testExternalGraphDriver("test-external-graph-driver", "spec", c)
s.testExternalGraphDriver("json-external-graph-driver", "json", c)
}
func (s *DockerExternalGraphdriverSuite) testExternalGraphDriver(name string, ext string, c *check.C) { |
c502fb49 |
s.d.StartWithBusybox(c, "-s", name) |
b78e4216 |
|
1f266d40 |
out, err := s.d.Cmd("run", "--name=graphtest", "busybox", "sh", "-c", "echo hello > /hello") |
b78e4216 |
c.Assert(err, check.IsNil, check.Commentf(out))
|
c502fb49 |
s.d.Restart(c, "-s", name) |
b78e4216 |
|
fab2a3dc |
out, err = s.d.Cmd("inspect", "--format={{.GraphDriver.Name}}", "graphtest") |
b78e4216 |
c.Assert(err, check.IsNil, check.Commentf(out)) |
de806a67 |
c.Assert(strings.TrimSpace(out), check.Equals, name) |
b78e4216 |
out, err = s.d.Cmd("diff", "graphtest")
c.Assert(err, check.IsNil, check.Commentf(out)) |
644a7426 |
c.Assert(strings.Contains(out, "A /hello"), check.Equals, true, check.Commentf("diff output: %s", out)) |
b78e4216 |
out, err = s.d.Cmd("rm", "-f", "graphtest")
c.Assert(err, check.IsNil, check.Commentf(out))
out, err = s.d.Cmd("info")
c.Assert(err, check.IsNil, check.Commentf(out))
|
c502fb49 |
s.d.Stop(c) |
b78e4216 |
|
4352da78 |
// Don't check s.ec.exists, because the daemon no longer calls the
// Exists function. |
de806a67 |
c.Assert(s.ec[ext].activations, check.Equals, 2)
c.Assert(s.ec[ext].init, check.Equals, 2)
c.Assert(s.ec[ext].creations >= 1, check.Equals, true)
c.Assert(s.ec[ext].removals >= 1, check.Equals, true)
c.Assert(s.ec[ext].gets >= 1, check.Equals, true)
c.Assert(s.ec[ext].puts >= 1, check.Equals, true) |
3343d234 |
c.Assert(s.ec[ext].stats, check.Equals, 5) |
de806a67 |
c.Assert(s.ec[ext].cleanups, check.Equals, 2)
c.Assert(s.ec[ext].applydiff >= 1, check.Equals, true)
c.Assert(s.ec[ext].changes, check.Equals, 1)
c.Assert(s.ec[ext].diffsize, check.Equals, 0)
c.Assert(s.ec[ext].diff, check.Equals, 0)
c.Assert(s.ec[ext].metadata, check.Equals, 1) |
b78e4216 |
}
func (s *DockerExternalGraphdriverSuite) TestExternalGraphDriverPull(c *check.C) { |
7781a1bf |
testRequires(c, Network, ExperimentalDaemon)
|
c502fb49 |
s.d.Start(c) |
b78e4216 |
out, err := s.d.Cmd("pull", "busybox:latest")
c.Assert(err, check.IsNil, check.Commentf(out))
out, err = s.d.Cmd("run", "-d", "busybox", "top")
c.Assert(err, check.IsNil, check.Commentf(out))
} |