func (d *WindowsGraphDriver) Import(id string, layerData archive.Reader, parentLayerPaths []string) (size int64, err error) { layerFs, err := d.Get(id, "") if err != nil { return } defer func() { if err != nil { d.Put(id) } }() tempFolder := layerFs + "-temp" if err = os.MkdirAll(tempFolder, 0755); err != nil { logrus.Errorf("Could not create %s %s", tempFolder, err) return } defer func() { if err2 := os.RemoveAll(tempFolder); err2 != nil { logrus.Warnf("Couldn't clean-up tempFolder: %s %s", tempFolder, err2) } }() start := time.Now().UTC() logrus.Debugf("Start untar layer") if size, err = chrootarchive.ApplyLayer(tempFolder, layerData); err != nil { return } logrus.Debugf("Untar time: %vs", time.Now().UTC().Sub(start).Seconds()) if err = hcsshim.ImportLayer(d.info, id, tempFolder, parentLayerPaths); err != nil { return } return }
// importLayer adds a new layer to the tag and graph store based on the given data. func (d *Driver) importLayer(id string, layerData archive.Reader, parentLayerPaths []string) (size int64, err error) { layerFolder := d.dir(id) tempFolder := layerFolder + "-" + strconv.FormatUint(uint64(random.Rand.Uint32()), 10) if err = os.MkdirAll(tempFolder, 0755); err != nil { logrus.Errorf("Could not create %s %s", tempFolder, err) return } defer func() { _, folderName := filepath.Split(tempFolder) if err2 := hcsshim.DestroyLayer(d.info, folderName); err2 != nil { logrus.Warnf("Couldn't clean-up tempFolder: %s %s", tempFolder, err2) } }() start := time.Now().UTC() logrus.Debugf("Start untar layer") if size, err = chrootarchive.ApplyLayer(tempFolder, layerData); err != nil { return } logrus.Debugf("Untar time: %vs", time.Now().UTC().Sub(start).Seconds()) if err = hcsshim.ImportLayer(d.info, id, tempFolder, parentLayerPaths); err != nil { return } return }
// ApplyDiff extracts the changeset from the given diff into the // layer with the specified id and parent, returning the size of the // new layer in bytes. func (gdw *naiveDiffDriver) ApplyDiff(id, parent string, diff archive.ArchiveReader) (bytes int64, err error) { driver := gdw.ProtoDriver // Mount the root filesystem so we can apply the diff/layer. layerFs, err := driver.Get(id, "") if err != nil { return } defer driver.Put(id) start := time.Now().UTC() log.Debugf("Start untar layer") if err = chrootarchive.ApplyLayer(layerFs, diff); err != nil { return } log.Debugf("Untar time: %vs", time.Now().UTC().Sub(start).Seconds()) if parent == "" { return utils.TreeSize(layerFs) } parentFs, err := driver.Get(parent, "") if err != nil { err = fmt.Errorf("Driver %s failed to get image parent %s: %s", driver, parent, err) return } defer driver.Put(parent) changes, err := archive.ChangesDirs(layerFs, parentFs) if err != nil { return } return archive.ChangesSize(layerFs, changes), nil }
func (d *Driver) ApplyDiff(id, parent string, diff archive.Reader) (size int64, err error) { // Mount the root filesystem so we can apply the diff/layer. layerFs, err := d.Get(id, "") if err != nil { return 0, err } defer d.Put(id) start := time.Now().UTC() glog.V(1).Info("Start untar layer") if size, err = chrootarchive.ApplyLayer(layerFs, diff); err != nil { return 0, err } glog.V(1).Infof("Untar time: %vs", time.Now().UTC().Sub(start).Seconds()) root := path.Join(utils.HYPER_ROOT, "vbox") idDisk := fmt.Sprintf("%s/images/%s.vdi", root, id) if _, err = os.Stat(idDisk); err != nil { return 0, err } if err = d.VmMountLayer(id); err != nil { return 0, err } // XXX should remove the image/container's directory return size, err }
// ApplyDiff applies the new layer on top of the root, if parent does not exist with will return a ErrApplyDiffFallback error. func (d *Driver) ApplyDiff(id string, parent string, diff archive.ArchiveReader) (size int64, err error) { dir := d.dir(id) if parent == "" { return 0, ErrApplyDiffFallback } parentRootDir := path.Join(d.dir(parent), "root") if _, err := os.Stat(parentRootDir); err != nil { return 0, ErrApplyDiffFallback } // We now know there is a parent, and it has a "root" directory containing // the full root filesystem. We can just hardlink it and apply the // layer. This relies on two things: // 1) ApplyDiff is only run once on a clean (no writes to upper layer) container // 2) ApplyDiff doesn't do any in-place writes to files (would break hardlinks) // These are all currently true and are not expected to break tmpRootDir, err := ioutil.TempDir(dir, "tmproot") if err != nil { return 0, err } defer func() { if err != nil { os.RemoveAll(tmpRootDir) } else { os.RemoveAll(path.Join(dir, "upper")) os.RemoveAll(path.Join(dir, "work")) os.RemoveAll(path.Join(dir, "merged")) os.RemoveAll(path.Join(dir, "lower-id")) } }() if err = copyDir(parentRootDir, tmpRootDir, copyHardlink); err != nil { return 0, err } start := time.Now().UTC() log.Debugf("Start untar layer") if err = chrootarchive.ApplyLayer(tmpRootDir, diff); err != nil { return 0, err } log.Debugf("Untar time: %vs", time.Now().UTC().Sub(start).Seconds()) changes, err := archive.ChangesDirs(tmpRootDir, parentRootDir, nil) if err != nil { return 0, err } size = archive.ChangesSize(tmpRootDir, changes) rootDir := path.Join(dir, "root") if err := os.Rename(tmpRootDir, rootDir); err != nil { return 0, err } return }
// WritePullData extracts manifest and rootfs to the disk. func WritePullData(pd PullData, dest string, extract bool) error { config, err := pd.Config() if err != nil { return err } var p types.Plugin if err := json.Unmarshal(config, &p); err != nil { return err } logrus.Debugf("%#v", p) if err := os.MkdirAll(dest, 0700); err != nil { return err } if extract { if err := ioutil.WriteFile(filepath.Join(dest, "manifest.json"), config, 0600); err != nil { return err } if err := os.MkdirAll(filepath.Join(dest, "rootfs"), 0700); err != nil { return err } } for i := 0; ; i++ { l, err := pd.Layer() if err == io.EOF { break } if err != nil { return err } if !extract { f, err := os.Create(filepath.Join(dest, fmt.Sprintf("layer%d.tar", i))) if err != nil { l.Close() return err } io.Copy(f, l) l.Close() f.Close() continue } if _, err := archive.ApplyLayer(filepath.Join(dest, "rootfs"), l); err != nil { return err } } return nil }
// ApplyDiff extracts the changeset from the given diff into the // layer with the specified id and parent, returning the size of the // new layer in bytes. func (d *WindowsGraphDriver) ApplyDiff(id, parent string, diff archive.Reader) (size int64, err error) { start := time.Now().UTC() logrus.Debugf("WindowsGraphDriver ApplyDiff: Start untar layer") destination := d.dir(id) if d.info.Flavour == diffDriver { destination = filepath.Dir(destination) } if size, err = chrootarchive.ApplyLayer(destination, diff); err != nil { return } logrus.Debugf("WindowsGraphDriver ApplyDiff: Untar time: %vs", time.Now().UTC().Sub(start).Seconds()) return }
// ApplyDiff extracts the changeset from the given diff into the // layer with the specified id and parent, returning the size of the // new layer in bytes. func (gdw *naiveDiffDriver) ApplyDiff(id, parent string, diff archive.ArchiveReader) (size int64, err error) { driver := gdw.ProtoDriver // Mount the root filesystem so we can apply the diff/layer. layerFs, err := driver.Get(id, "") if err != nil { return } defer driver.Put(id) start := time.Now().UTC() logrus.Debugf("Start untar layer") if size, err = chrootarchive.ApplyLayer(layerFs, diff); err != nil { return } logrus.Debugf("Untar time: %vs", time.Now().UTC().Sub(start).Seconds()) return }
// importLayer adds a new layer to the tag and graph store based on the given data. func (d *Driver) importLayer(id string, layerData archive.Reader, parentLayerPaths []string) (size int64, err error) { if hcsshim.IsTP4() { // Import from TP4 format to maintain compatibility with existing images. var tempFolder string tempFolder, err = ioutil.TempDir("", "hcs") if err != nil { return } defer os.RemoveAll(tempFolder) if size, err = chrootarchive.ApplyLayer(tempFolder, layerData); err != nil { return } if err = hcsshim.ImportLayer(d.info, id, tempFolder, parentLayerPaths); err != nil { return } return } var w hcsshim.LayerWriter w, err = hcsshim.NewLayerWriter(d.info, id, parentLayerPaths) if err != nil { return } size, err = writeLayerFromTar(layerData, w) if err != nil { w.Close() return } err = w.Close() if err != nil { return } return }