integration-cli/docker_cli_external_graphdriver_unix_test.go
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))
 }