123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462 |
- package graphdriver
- import (
- "context"
- "encoding/json"
- "fmt"
- "io"
- "net/http"
- "net/http/httptest"
- "os"
- "runtime"
- "testing"
- "github.com/docker/docker/api/types"
- containertypes "github.com/docker/docker/api/types/container"
- "github.com/docker/docker/client"
- "github.com/docker/docker/daemon/graphdriver"
- "github.com/docker/docker/daemon/graphdriver/vfs"
- "github.com/docker/docker/integration/internal/container"
- "github.com/docker/docker/integration/internal/requirement"
- "github.com/docker/docker/pkg/archive"
- "github.com/docker/docker/pkg/idtools"
- "github.com/docker/docker/pkg/plugins"
- "github.com/docker/docker/testutil/daemon"
- "gotest.tools/v3/assert"
- is "gotest.tools/v3/assert/cmp"
- "gotest.tools/v3/skip"
- )
- 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 TestExternalGraphDriver(t *testing.T) {
- skip.If(t, runtime.GOOS == "windows")
- skip.If(t, testEnv.IsRemoteDaemon, "cannot run daemon when remote daemon")
- skip.If(t, !requirement.HasHubConnectivity(t))
- skip.If(t, testEnv.IsRootless, "rootless mode doesn't support external graph driver")
- // Setup plugin(s)
- ec := make(map[string]*graphEventsCounter)
- sserver := setupPluginViaSpecFile(t, ec)
- jserver := setupPluginViaJSONFile(t, ec)
- // Create daemon
- d := daemon.New(t, daemon.WithExperimental())
- c := d.NewClientT(t)
- for _, tc := range []struct {
- name string
- test func(client.APIClient, *daemon.Daemon) func(*testing.T)
- }{
- {
- name: "json",
- test: testExternalGraphDriver("json", ec),
- },
- {
- name: "spec",
- test: testExternalGraphDriver("spec", ec),
- },
- {
- name: "pull",
- test: testGraphDriverPull,
- },
- } {
- t.Run(tc.name, tc.test(c, d))
- }
- sserver.Close()
- jserver.Close()
- err := os.RemoveAll("/etc/docker/plugins")
- assert.NilError(t, err)
- }
- func setupPluginViaSpecFile(t *testing.T, ec map[string]*graphEventsCounter) *httptest.Server {
- mux := http.NewServeMux()
- server := httptest.NewServer(mux)
- setupPlugin(t, ec, "spec", mux, []byte(server.URL))
- return server
- }
- func setupPluginViaJSONFile(t *testing.T, ec map[string]*graphEventsCounter) *httptest.Server {
- mux := http.NewServeMux()
- server := httptest.NewServer(mux)
- p := plugins.NewLocalPlugin("json-external-graph-driver", server.URL)
- b, err := json.Marshal(p)
- assert.NilError(t, err)
- setupPlugin(t, ec, "json", mux, b)
- return server
- }
- func setupPlugin(t *testing.T, ec map[string]*graphEventsCounter, ext string, mux *http.ServeMux, b []byte) {
- name := fmt.Sprintf("%s-external-graph-driver", ext)
- type graphDriverRequest struct {
- ID string `json:",omitempty"`
- Parent string `json:",omitempty"`
- MountLabel string `json:",omitempty"`
- ReadOnly bool `json:",omitempty"`
- }
- 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{}) {
- w.Header().Set("Content-Type", "application/vnd.docker.plugins.v1+json")
- switch t := data.(type) {
- case error:
- fmt.Fprintf(w, "{\"Err\": %q}\n", t.Error())
- case string:
- fmt.Fprintln(w, t)
- default:
- json.NewEncoder(w).Encode(&data)
- }
- }
- 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
- }
- base, err := os.MkdirTemp("", name)
- assert.NilError(t, err)
- vfsProto, err := vfs.Init(base, []string{}, idtools.IdentityMapping{})
- assert.NilError(t, err, "error initializing graph driver")
- driver := graphdriver.NewNaiveDiffDriver(vfsProto, idtools.IdentityMapping{})
- ec[ext] = &graphEventsCounter{}
- mux.HandleFunc("/Plugin.Activate", func(w http.ResponseWriter, r *http.Request) {
- ec[ext].activations++
- respond(w, `{"Implements": ["GraphDriver"]}`)
- })
- mux.HandleFunc("/GraphDriver.Init", func(w http.ResponseWriter, r *http.Request) {
- ec[ext].init++
- respond(w, "{}")
- })
- mux.HandleFunc("/GraphDriver.CreateReadWrite", func(w http.ResponseWriter, r *http.Request) {
- ec[ext].creations++
- var req graphDriverRequest
- if err := decReq(r.Body, &req, w); err != nil {
- return
- }
- if err := driver.CreateReadWrite(req.ID, req.Parent, nil); err != nil {
- respond(w, err)
- return
- }
- respond(w, "{}")
- })
- mux.HandleFunc("/GraphDriver.Create", func(w http.ResponseWriter, r *http.Request) {
- ec[ext].creations++
- var req graphDriverRequest
- if err := decReq(r.Body, &req, w); err != nil {
- return
- }
- if err := driver.Create(req.ID, req.Parent, nil); err != nil {
- respond(w, err)
- return
- }
- respond(w, "{}")
- })
- mux.HandleFunc("/GraphDriver.Remove", func(w http.ResponseWriter, r *http.Request) {
- ec[ext].removals++
- var req graphDriverRequest
- if err := decReq(r.Body, &req, w); err != nil {
- 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) {
- ec[ext].gets++
- var req graphDriverRequest
- if err := decReq(r.Body, &req, w); err != nil {
- return
- }
- 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) {
- ec[ext].puts++
- var req graphDriverRequest
- if err := decReq(r.Body, &req, w); err != nil {
- 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) {
- ec[ext].exists++
- var req graphDriverRequest
- if err := decReq(r.Body, &req, w); err != nil {
- return
- }
- respond(w, &graphDriverResponse{Exists: driver.Exists(req.ID)})
- })
- mux.HandleFunc("/GraphDriver.Status", func(w http.ResponseWriter, r *http.Request) {
- ec[ext].stats++
- respond(w, &graphDriverResponse{Status: driver.Status()})
- })
- mux.HandleFunc("/GraphDriver.Cleanup", func(w http.ResponseWriter, r *http.Request) {
- ec[ext].cleanups++
- err := driver.Cleanup()
- if err != nil {
- respond(w, err)
- return
- }
- respond(w, `{}`)
- })
- mux.HandleFunc("/GraphDriver.GetMetadata", func(w http.ResponseWriter, r *http.Request) {
- ec[ext].metadata++
- var req graphDriverRequest
- if err := decReq(r.Body, &req, w); err != nil {
- 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) {
- ec[ext].diff++
- var req graphDriverRequest
- if err := decReq(r.Body, &req, w); err != nil {
- 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) {
- ec[ext].changes++
- var req graphDriverRequest
- if err := decReq(r.Body, &req, w); err != nil {
- 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) {
- ec[ext].applydiff++
- diff := r.Body
- defer r.Body.Close()
- id := r.URL.Query().Get("id")
- parent := r.URL.Query().Get("parent")
- if id == "" {
- http.Error(w, "missing id", 409)
- }
- size, err := driver.ApplyDiff(id, parent, diff)
- if err != nil {
- respond(w, err)
- return
- }
- respond(w, &graphDriverResponse{Size: size})
- })
- mux.HandleFunc("/GraphDriver.DiffSize", func(w http.ResponseWriter, r *http.Request) {
- ec[ext].diffsize++
- var req graphDriverRequest
- if err := decReq(r.Body, &req, w); err != nil {
- return
- }
- size, err := driver.DiffSize(req.ID, req.Parent)
- if err != nil {
- respond(w, err)
- return
- }
- respond(w, &graphDriverResponse{Size: size})
- })
- err = os.MkdirAll("/etc/docker/plugins", 0755)
- assert.NilError(t, err)
- specFile := "/etc/docker/plugins/" + name + "." + ext
- err = os.WriteFile(specFile, b, 0644)
- assert.NilError(t, err)
- }
- func testExternalGraphDriver(ext string, ec map[string]*graphEventsCounter) func(client.APIClient, *daemon.Daemon) func(*testing.T) {
- return func(c client.APIClient, d *daemon.Daemon) func(*testing.T) {
- return func(t *testing.T) {
- driverName := fmt.Sprintf("%s-external-graph-driver", ext)
- d.StartWithBusybox(t, "-s", driverName)
- ctx := context.Background()
- testGraphDriver(ctx, t, c, driverName, func(t *testing.T) {
- d.Restart(t, "-s", driverName)
- })
- _, err := c.Info(ctx)
- assert.NilError(t, err)
- d.Stop(t)
- // Don't check ec.exists, because the daemon no longer calls the
- // Exists function.
- assert.Check(t, is.Equal(ec[ext].activations, 2))
- assert.Check(t, is.Equal(ec[ext].init, 2))
- assert.Check(t, ec[ext].creations >= 1)
- assert.Check(t, ec[ext].removals >= 1)
- assert.Check(t, ec[ext].gets >= 1)
- assert.Check(t, ec[ext].puts >= 1)
- assert.Check(t, is.Equal(ec[ext].stats, 5))
- assert.Check(t, is.Equal(ec[ext].cleanups, 2))
- assert.Check(t, ec[ext].applydiff >= 1)
- assert.Check(t, is.Equal(ec[ext].changes, 1))
- assert.Check(t, is.Equal(ec[ext].diffsize, 0))
- assert.Check(t, is.Equal(ec[ext].diff, 0))
- assert.Check(t, is.Equal(ec[ext].metadata, 1))
- }
- }
- }
- func testGraphDriverPull(c client.APIClient, d *daemon.Daemon) func(*testing.T) {
- return func(t *testing.T) {
- d.Start(t)
- defer d.Stop(t)
- ctx := context.Background()
- r, err := c.ImagePull(ctx, "busybox:latest@sha256:95cf004f559831017cdf4628aaf1bb30133677be8702a8c5f2994629f637a209", types.ImagePullOptions{})
- assert.NilError(t, err)
- _, err = io.Copy(io.Discard, r)
- assert.NilError(t, err)
- container.Run(ctx, t, c, container.WithImage("busybox:latest@sha256:95cf004f559831017cdf4628aaf1bb30133677be8702a8c5f2994629f637a209"))
- }
- }
- func TestGraphdriverPluginV2(t *testing.T) {
- skip.If(t, runtime.GOOS == "windows")
- skip.If(t, testEnv.IsRemoteDaemon, "cannot run daemon when remote daemon")
- skip.If(t, !requirement.HasHubConnectivity(t))
- skip.If(t, os.Getenv("DOCKER_ENGINE_GOARCH") != "amd64")
- skip.If(t, !requirement.Overlay2Supported(testEnv.DaemonInfo.KernelVersion))
- d := daemon.New(t, daemon.WithExperimental())
- d.Start(t)
- defer d.Stop(t)
- client := d.NewClientT(t)
- defer client.Close()
- ctx := context.Background()
- // install the plugin
- plugin := "cpuguy83/docker-overlay2-graphdriver-plugin"
- responseReader, err := client.PluginInstall(ctx, plugin, types.PluginInstallOptions{
- RemoteRef: plugin,
- AcceptAllPermissions: true,
- })
- assert.NilError(t, err)
- defer responseReader.Close()
- // ensure it's done by waiting for EOF on the response
- _, err = io.Copy(io.Discard, responseReader)
- assert.NilError(t, err)
- // restart the daemon with the plugin set as the storage driver
- d.Stop(t)
- d.StartWithBusybox(t, "-s", plugin)
- testGraphDriver(ctx, t, client, plugin, nil)
- }
- func testGraphDriver(ctx context.Context, t *testing.T, c client.APIClient, driverName string, afterContainerRunFn func(*testing.T)) {
- id := container.Run(ctx, t, c, container.WithCmd("sh", "-c", "echo hello > /hello"))
- if afterContainerRunFn != nil {
- afterContainerRunFn(t)
- }
- i, err := c.ContainerInspect(ctx, id)
- assert.NilError(t, err)
- assert.Check(t, is.Equal(i.GraphDriver.Name, driverName))
- diffs, err := c.ContainerDiff(ctx, id)
- assert.NilError(t, err)
- assert.Check(t, is.Contains(diffs, containertypes.ContainerChangeResponseItem{
- Kind: archive.ChangeAdd,
- Path: "/hello",
- }), "diffs: %v", diffs)
- err = c.ContainerRemove(ctx, id, types.ContainerRemoveOptions{
- Force: true,
- })
- assert.NilError(t, err)
- }
|